Blame view

drivers/dma-buf/dma-fence.c 18.6 KB
1802d0bee   Thomas Gleixner   treewide: Replace...
1
  // SPDX-License-Identifier: GPL-2.0-only
e941759c7   Maarten Lankhorst   fence: dma-buf cr...
2
3
4
5
6
7
8
9
10
  /*
   * Fence mechanism for dma-buf and to allow for asynchronous dma access
   *
   * Copyright (C) 2012 Canonical Ltd
   * Copyright (C) 2012 Texas Instruments
   *
   * Authors:
   * Rob Clark <robdclark@gmail.com>
   * Maarten Lankhorst <maarten.lankhorst@canonical.com>
e941759c7   Maarten Lankhorst   fence: dma-buf cr...
11
12
13
14
15
   */
  
  #include <linux/slab.h>
  #include <linux/export.h>
  #include <linux/atomic.h>
f54d18670   Chris Wilson   dma-buf: Rename s...
16
  #include <linux/dma-fence.h>
174cd4b1e   Ingo Molnar   sched/headers: Pr...
17
  #include <linux/sched/signal.h>
e941759c7   Maarten Lankhorst   fence: dma-buf cr...
18
19
  
  #define CREATE_TRACE_POINTS
f54d18670   Chris Wilson   dma-buf: Rename s...
20
  #include <trace/events/dma_fence.h>
e941759c7   Maarten Lankhorst   fence: dma-buf cr...
21

f54d18670   Chris Wilson   dma-buf: Rename s...
22
  EXPORT_TRACEPOINT_SYMBOL(dma_fence_emit);
8c96c6780   Chris Wilson   dma/fence: Export...
23
  EXPORT_TRACEPOINT_SYMBOL(dma_fence_enable_signal);
c36beba6b   Chris Wilson   drm/i915: Seal ra...
24
  EXPORT_TRACEPOINT_SYMBOL(dma_fence_signaled);
e941759c7   Maarten Lankhorst   fence: dma-buf cr...
25

078dec332   Christian König   dma-buf: add dma_...
26
27
  static DEFINE_SPINLOCK(dma_fence_stub_lock);
  static struct dma_fence dma_fence_stub;
e9f3b7964   Thierry Reding   dma-buf/fence: Fi...
28
  /*
e941759c7   Maarten Lankhorst   fence: dma-buf cr...
29
30
31
32
33
   * fence context counter: each execution context should have its own
   * fence context, this allows checking if fences belong to the same
   * context or not. One device can have multiple separate contexts,
   * and they're used if some engine can run independently of another.
   */
078dec332   Christian König   dma-buf: add dma_...
34
  static atomic64_t dma_fence_context_counter = ATOMIC64_INIT(1);
e941759c7   Maarten Lankhorst   fence: dma-buf cr...
35
36
  
  /**
4dd3cdb28   Daniel Vetter   dma-fence: Polish...
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
   * DOC: DMA fences overview
   *
   * DMA fences, represented by &struct dma_fence, are the kernel internal
   * synchronization primitive for DMA operations like GPU rendering, video
   * encoding/decoding, or displaying buffers on a screen.
   *
   * A fence is initialized using dma_fence_init() and completed using
   * dma_fence_signal(). Fences are associated with a context, allocated through
   * dma_fence_context_alloc(), and all fences on the same context are
   * fully ordered.
   *
   * Since the purposes of fences is to facilitate cross-device and
   * cross-application synchronization, there's multiple ways to use one:
   *
   * - Individual fences can be exposed as a &sync_file, accessed as a file
   *   descriptor from userspace, created by calling sync_file_create(). This is
   *   called explicit fencing, since userspace passes around explicit
   *   synchronization points.
   *
   * - Some subsystems also have their own explicit fencing primitives, like
   *   &drm_syncobj. Compared to &sync_file, a &drm_syncobj allows the underlying
   *   fence to be updated.
   *
   * - Then there's also implicit fencing, where the synchronization points are
   *   implicitly passed around as part of shared &dma_buf instances. Such
52791eeec   Christian König   dma-buf: rename r...
62
   *   implicit fences are stored in &struct dma_resv through the
4dd3cdb28   Daniel Vetter   dma-fence: Polish...
63
64
   *   &dma_buf.resv pointer.
   */
078dec332   Christian König   dma-buf: add dma_...
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
  static const char *dma_fence_stub_get_name(struct dma_fence *fence)
  {
          return "stub";
  }
  
  static const struct dma_fence_ops dma_fence_stub_ops = {
  	.get_driver_name = dma_fence_stub_get_name,
  	.get_timeline_name = dma_fence_stub_get_name,
  };
  
  /**
   * dma_fence_get_stub - return a signaled fence
   *
   * Return a stub fence which is already signaled.
   */
  struct dma_fence *dma_fence_get_stub(void)
  {
  	spin_lock(&dma_fence_stub_lock);
  	if (!dma_fence_stub.ops) {
  		dma_fence_init(&dma_fence_stub,
  			       &dma_fence_stub_ops,
  			       &dma_fence_stub_lock,
  			       0, 0);
  		dma_fence_signal_locked(&dma_fence_stub);
  	}
  	spin_unlock(&dma_fence_stub_lock);
  
  	return dma_fence_get(&dma_fence_stub);
  }
  EXPORT_SYMBOL(dma_fence_get_stub);
4dd3cdb28   Daniel Vetter   dma-fence: Polish...
95
  /**
f54d18670   Chris Wilson   dma-buf: Rename s...
96
   * dma_fence_context_alloc - allocate an array of fence contexts
4dd3cdb28   Daniel Vetter   dma-fence: Polish...
97
   * @num: amount of contexts to allocate
e941759c7   Maarten Lankhorst   fence: dma-buf cr...
98
   *
4dd3cdb28   Daniel Vetter   dma-fence: Polish...
99
100
101
   * This function will return the first index of the number of fence contexts
   * allocated.  The fence context is used for setting &dma_fence.context to a
   * unique number by passing the context to dma_fence_init().
e941759c7   Maarten Lankhorst   fence: dma-buf cr...
102
   */
f54d18670   Chris Wilson   dma-buf: Rename s...
103
  u64 dma_fence_context_alloc(unsigned num)
e941759c7   Maarten Lankhorst   fence: dma-buf cr...
104
  {
6ce31263c   Daniel Vetter   dma-fence: Don't ...
105
  	WARN_ON(!num);
f54d18670   Chris Wilson   dma-buf: Rename s...
106
  	return atomic64_add_return(num, &dma_fence_context_counter) - num;
e941759c7   Maarten Lankhorst   fence: dma-buf cr...
107
  }
f54d18670   Chris Wilson   dma-buf: Rename s...
108
  EXPORT_SYMBOL(dma_fence_context_alloc);
e941759c7   Maarten Lankhorst   fence: dma-buf cr...
109
110
  
  /**
f54d18670   Chris Wilson   dma-buf: Rename s...
111
   * dma_fence_signal_locked - signal completion of a fence
e941759c7   Maarten Lankhorst   fence: dma-buf cr...
112
113
114
   * @fence: the fence to signal
   *
   * Signal completion for software callbacks on a fence, this will unblock
f54d18670   Chris Wilson   dma-buf: Rename s...
115
116
   * dma_fence_wait() calls and run all the callbacks added with
   * dma_fence_add_callback(). Can be called multiple times, but since a fence
4dd3cdb28   Daniel Vetter   dma-fence: Polish...
117
118
119
120
121
   * can only go from the unsignaled to the signaled state and not back, it will
   * only be effective the first time.
   *
   * Unlike dma_fence_signal(), this function must be called with &dma_fence.lock
   * held.
e941759c7   Maarten Lankhorst   fence: dma-buf cr...
122
   *
4dd3cdb28   Daniel Vetter   dma-fence: Polish...
123
124
   * Returns 0 on success and a negative error value when @fence has been
   * signalled already.
e941759c7   Maarten Lankhorst   fence: dma-buf cr...
125
   */
f54d18670   Chris Wilson   dma-buf: Rename s...
126
  int dma_fence_signal_locked(struct dma_fence *fence)
e941759c7   Maarten Lankhorst   fence: dma-buf cr...
127
  {
f54d18670   Chris Wilson   dma-buf: Rename s...
128
  	struct dma_fence_cb *cur, *tmp;
f2cb60e9a   Chris Wilson   dma-fence: Store ...
129
  	struct list_head cb_list;
e941759c7   Maarten Lankhorst   fence: dma-buf cr...
130

78010cd97   Rob Clark   dma-buf/fence: ad...
131
  	lockdep_assert_held(fence->lock);
0fc89b680   Chris Wilson   dma-fence: Simply...
132
133
  	if (unlikely(test_and_set_bit(DMA_FENCE_FLAG_SIGNALED_BIT,
  				      &fence->flags)))
e941759c7   Maarten Lankhorst   fence: dma-buf cr...
134
  		return -EINVAL;
f2cb60e9a   Chris Wilson   dma-fence: Store ...
135
136
  	/* Stash the cb_list before replacing it with the timestamp */
  	list_replace(&fence->cb_list, &cb_list);
0fc89b680   Chris Wilson   dma-fence: Simply...
137
138
139
  	fence->timestamp = ktime_get();
  	set_bit(DMA_FENCE_FLAG_TIMESTAMP_BIT, &fence->flags);
  	trace_dma_fence_signaled(fence);
e941759c7   Maarten Lankhorst   fence: dma-buf cr...
140

f2cb60e9a   Chris Wilson   dma-fence: Store ...
141
142
143
  	list_for_each_entry_safe(cur, tmp, &cb_list, node) {
  		INIT_LIST_HEAD(&cur->node);
  		cur->func(fence, cur);
e941759c7   Maarten Lankhorst   fence: dma-buf cr...
144
  	}
0fc89b680   Chris Wilson   dma-fence: Simply...
145
146
  
  	return 0;
e941759c7   Maarten Lankhorst   fence: dma-buf cr...
147
  }
f54d18670   Chris Wilson   dma-buf: Rename s...
148
  EXPORT_SYMBOL(dma_fence_signal_locked);
e941759c7   Maarten Lankhorst   fence: dma-buf cr...
149
150
  
  /**
f54d18670   Chris Wilson   dma-buf: Rename s...
151
   * dma_fence_signal - signal completion of a fence
e941759c7   Maarten Lankhorst   fence: dma-buf cr...
152
153
154
   * @fence: the fence to signal
   *
   * Signal completion for software callbacks on a fence, this will unblock
f54d18670   Chris Wilson   dma-buf: Rename s...
155
156
   * dma_fence_wait() calls and run all the callbacks added with
   * dma_fence_add_callback(). Can be called multiple times, but since a fence
4dd3cdb28   Daniel Vetter   dma-fence: Polish...
157
158
159
160
161
   * can only go from the unsignaled to the signaled state and not back, it will
   * only be effective the first time.
   *
   * Returns 0 on success and a negative error value when @fence has been
   * signalled already.
e941759c7   Maarten Lankhorst   fence: dma-buf cr...
162
   */
f54d18670   Chris Wilson   dma-buf: Rename s...
163
  int dma_fence_signal(struct dma_fence *fence)
e941759c7   Maarten Lankhorst   fence: dma-buf cr...
164
165
  {
  	unsigned long flags;
0fc89b680   Chris Wilson   dma-fence: Simply...
166
  	int ret;
e941759c7   Maarten Lankhorst   fence: dma-buf cr...
167
168
169
  
  	if (!fence)
  		return -EINVAL;
0fc89b680   Chris Wilson   dma-fence: Simply...
170
171
172
  	spin_lock_irqsave(fence->lock, flags);
  	ret = dma_fence_signal_locked(fence);
  	spin_unlock_irqrestore(fence->lock, flags);
e941759c7   Maarten Lankhorst   fence: dma-buf cr...
173

0fc89b680   Chris Wilson   dma-fence: Simply...
174
  	return ret;
e941759c7   Maarten Lankhorst   fence: dma-buf cr...
175
  }
f54d18670   Chris Wilson   dma-buf: Rename s...
176
  EXPORT_SYMBOL(dma_fence_signal);
e941759c7   Maarten Lankhorst   fence: dma-buf cr...
177
178
  
  /**
f54d18670   Chris Wilson   dma-buf: Rename s...
179
   * dma_fence_wait_timeout - sleep until the fence gets signaled
e941759c7   Maarten Lankhorst   fence: dma-buf cr...
180
   * or until timeout elapses
4dd3cdb28   Daniel Vetter   dma-fence: Polish...
181
182
183
   * @fence: the fence to wait on
   * @intr: if true, do an interruptible wait
   * @timeout: timeout value in jiffies, or MAX_SCHEDULE_TIMEOUT
e941759c7   Maarten Lankhorst   fence: dma-buf cr...
184
185
186
187
188
189
190
191
192
   *
   * Returns -ERESTARTSYS if interrupted, 0 if the wait timed out, or the
   * remaining timeout in jiffies on success. Other error values may be
   * returned on custom implementations.
   *
   * Performs a synchronous wait on this fence. It is assumed the caller
   * directly or indirectly (buf-mgr between reservation and committing)
   * holds a reference to the fence, otherwise the fence might be
   * freed before return, resulting in undefined behavior.
4dd3cdb28   Daniel Vetter   dma-fence: Polish...
193
194
   *
   * See also dma_fence_wait() and dma_fence_wait_any_timeout().
e941759c7   Maarten Lankhorst   fence: dma-buf cr...
195
196
   */
  signed long
f54d18670   Chris Wilson   dma-buf: Rename s...
197
  dma_fence_wait_timeout(struct dma_fence *fence, bool intr, signed long timeout)
e941759c7   Maarten Lankhorst   fence: dma-buf cr...
198
199
200
201
202
  {
  	signed long ret;
  
  	if (WARN_ON(timeout < 0))
  		return -EINVAL;
f54d18670   Chris Wilson   dma-buf: Rename s...
203
  	trace_dma_fence_wait_start(fence);
418cc6ca0   Daniel Vetter   dma-fence: Make -...
204
205
206
207
  	if (fence->ops->wait)
  		ret = fence->ops->wait(fence, intr, timeout);
  	else
  		ret = dma_fence_default_wait(fence, intr, timeout);
f54d18670   Chris Wilson   dma-buf: Rename s...
208
  	trace_dma_fence_wait_end(fence);
e941759c7   Maarten Lankhorst   fence: dma-buf cr...
209
210
  	return ret;
  }
f54d18670   Chris Wilson   dma-buf: Rename s...
211
  EXPORT_SYMBOL(dma_fence_wait_timeout);
e941759c7   Maarten Lankhorst   fence: dma-buf cr...
212

4dd3cdb28   Daniel Vetter   dma-fence: Polish...
213
214
215
216
217
218
219
  /**
   * dma_fence_release - default relese function for fences
   * @kref: &dma_fence.recfount
   *
   * This is the default release functions for &dma_fence. Drivers shouldn't call
   * this directly, but instead call dma_fence_put().
   */
f54d18670   Chris Wilson   dma-buf: Rename s...
220
  void dma_fence_release(struct kref *kref)
e941759c7   Maarten Lankhorst   fence: dma-buf cr...
221
  {
f54d18670   Chris Wilson   dma-buf: Rename s...
222
223
  	struct dma_fence *fence =
  		container_of(kref, struct dma_fence, refcount);
e941759c7   Maarten Lankhorst   fence: dma-buf cr...
224

f54d18670   Chris Wilson   dma-buf: Rename s...
225
  	trace_dma_fence_destroy(fence);
e941759c7   Maarten Lankhorst   fence: dma-buf cr...
226

f2cb60e9a   Chris Wilson   dma-fence: Store ...
227
228
  	if (WARN(!list_empty(&fence->cb_list) &&
  		 !test_bit(DMA_FENCE_FLAG_SIGNALED_BIT, &fence->flags),
427231bc6   Chris Wilson   dma-fence: Signal...
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
  		 "Fence %s:%s:%llx:%llx released with pending signals!
  ",
  		 fence->ops->get_driver_name(fence),
  		 fence->ops->get_timeline_name(fence),
  		 fence->context, fence->seqno)) {
  		unsigned long flags;
  
  		/*
  		 * Failed to signal before release, likely a refcounting issue.
  		 *
  		 * This should never happen, but if it does make sure that we
  		 * don't leave chains dangling. We set the error flag first
  		 * so that the callbacks know this signal is due to an error.
  		 */
  		spin_lock_irqsave(fence->lock, flags);
  		fence->error = -EDEADLK;
  		dma_fence_signal_locked(fence);
  		spin_unlock_irqrestore(fence->lock, flags);
  	}
e941759c7   Maarten Lankhorst   fence: dma-buf cr...
248
249
250
251
  
  	if (fence->ops->release)
  		fence->ops->release(fence);
  	else
f54d18670   Chris Wilson   dma-buf: Rename s...
252
  		dma_fence_free(fence);
e941759c7   Maarten Lankhorst   fence: dma-buf cr...
253
  }
f54d18670   Chris Wilson   dma-buf: Rename s...
254
  EXPORT_SYMBOL(dma_fence_release);
e941759c7   Maarten Lankhorst   fence: dma-buf cr...
255

4dd3cdb28   Daniel Vetter   dma-fence: Polish...
256
257
258
259
260
261
262
  /**
   * dma_fence_free - default release function for &dma_fence.
   * @fence: fence to release
   *
   * This is the default implementation for &dma_fence_ops.release. It calls
   * kfree_rcu() on @fence.
   */
f54d18670   Chris Wilson   dma-buf: Rename s...
263
  void dma_fence_free(struct dma_fence *fence)
e941759c7   Maarten Lankhorst   fence: dma-buf cr...
264
  {
3c3b177a9   Maarten Lankhorst   reservation: add ...
265
  	kfree_rcu(fence, rcu);
e941759c7   Maarten Lankhorst   fence: dma-buf cr...
266
  }
f54d18670   Chris Wilson   dma-buf: Rename s...
267
  EXPORT_SYMBOL(dma_fence_free);
e941759c7   Maarten Lankhorst   fence: dma-buf cr...
268
269
  
  /**
f54d18670   Chris Wilson   dma-buf: Rename s...
270
   * dma_fence_enable_sw_signaling - enable signaling on fence
4dd3cdb28   Daniel Vetter   dma-fence: Polish...
271
   * @fence: the fence to enable
e941759c7   Maarten Lankhorst   fence: dma-buf cr...
272
   *
4dd3cdb28   Daniel Vetter   dma-fence: Polish...
273
274
275
   * This will request for sw signaling to be enabled, to make the fence
   * complete as soon as possible. This calls &dma_fence_ops.enable_signaling
   * internally.
e941759c7   Maarten Lankhorst   fence: dma-buf cr...
276
   */
f54d18670   Chris Wilson   dma-buf: Rename s...
277
  void dma_fence_enable_sw_signaling(struct dma_fence *fence)
e941759c7   Maarten Lankhorst   fence: dma-buf cr...
278
279
  {
  	unsigned long flags;
f54d18670   Chris Wilson   dma-buf: Rename s...
280
281
  	if (!test_and_set_bit(DMA_FENCE_FLAG_ENABLE_SIGNAL_BIT,
  			      &fence->flags) &&
c701317a3   Daniel Vetter   dma-fence: Make -...
282
283
  	    !test_bit(DMA_FENCE_FLAG_SIGNALED_BIT, &fence->flags) &&
  	    fence->ops->enable_signaling) {
f54d18670   Chris Wilson   dma-buf: Rename s...
284
  		trace_dma_fence_enable_signal(fence);
e941759c7   Maarten Lankhorst   fence: dma-buf cr...
285
286
287
288
  
  		spin_lock_irqsave(fence->lock, flags);
  
  		if (!fence->ops->enable_signaling(fence))
f54d18670   Chris Wilson   dma-buf: Rename s...
289
  			dma_fence_signal_locked(fence);
e941759c7   Maarten Lankhorst   fence: dma-buf cr...
290
291
292
293
  
  		spin_unlock_irqrestore(fence->lock, flags);
  	}
  }
f54d18670   Chris Wilson   dma-buf: Rename s...
294
  EXPORT_SYMBOL(dma_fence_enable_sw_signaling);
e941759c7   Maarten Lankhorst   fence: dma-buf cr...
295
296
  
  /**
f54d18670   Chris Wilson   dma-buf: Rename s...
297
   * dma_fence_add_callback - add a callback to be called when the fence
e941759c7   Maarten Lankhorst   fence: dma-buf cr...
298
   * is signaled
4dd3cdb28   Daniel Vetter   dma-fence: Polish...
299
300
301
   * @fence: the fence to wait on
   * @cb: the callback to register
   * @func: the function to call
e941759c7   Maarten Lankhorst   fence: dma-buf cr...
302
   *
4dd3cdb28   Daniel Vetter   dma-fence: Polish...
303
   * @cb will be initialized by dma_fence_add_callback(), no initialization
e941759c7   Maarten Lankhorst   fence: dma-buf cr...
304
305
306
307
308
   * by the caller is required. Any number of callbacks can be registered
   * to a fence, but a callback can only be registered to one fence at a time.
   *
   * Note that the callback can be called from an atomic context.  If
   * fence is already signaled, this function will return -ENOENT (and
4dd3cdb28   Daniel Vetter   dma-fence: Polish...
309
   * *not* call the callback).
e941759c7   Maarten Lankhorst   fence: dma-buf cr...
310
311
   *
   * Add a software callback to the fence. Same restrictions apply to
4dd3cdb28   Daniel Vetter   dma-fence: Polish...
312
313
314
315
316
   * refcount as it does to dma_fence_wait(), however the caller doesn't need to
   * keep a refcount to fence afterward dma_fence_add_callback() has returned:
   * when software access is enabled, the creator of the fence is required to keep
   * the fence alive until after it signals with dma_fence_signal(). The callback
   * itself can be called from irq context.
e941759c7   Maarten Lankhorst   fence: dma-buf cr...
317
   *
f642de16c   Gustavo Padovan   dma-buf/dma-fence...
318
319
   * Returns 0 in case of success, -ENOENT if the fence is already signaled
   * and -EINVAL in case of error.
e941759c7   Maarten Lankhorst   fence: dma-buf cr...
320
   */
f54d18670   Chris Wilson   dma-buf: Rename s...
321
322
  int dma_fence_add_callback(struct dma_fence *fence, struct dma_fence_cb *cb,
  			   dma_fence_func_t func)
e941759c7   Maarten Lankhorst   fence: dma-buf cr...
323
324
325
326
327
328
329
  {
  	unsigned long flags;
  	int ret = 0;
  	bool was_set;
  
  	if (WARN_ON(!fence || !func))
  		return -EINVAL;
f54d18670   Chris Wilson   dma-buf: Rename s...
330
  	if (test_bit(DMA_FENCE_FLAG_SIGNALED_BIT, &fence->flags)) {
e941759c7   Maarten Lankhorst   fence: dma-buf cr...
331
332
333
334
335
  		INIT_LIST_HEAD(&cb->node);
  		return -ENOENT;
  	}
  
  	spin_lock_irqsave(fence->lock, flags);
f54d18670   Chris Wilson   dma-buf: Rename s...
336
337
  	was_set = test_and_set_bit(DMA_FENCE_FLAG_ENABLE_SIGNAL_BIT,
  				   &fence->flags);
e941759c7   Maarten Lankhorst   fence: dma-buf cr...
338

f54d18670   Chris Wilson   dma-buf: Rename s...
339
  	if (test_bit(DMA_FENCE_FLAG_SIGNALED_BIT, &fence->flags))
e941759c7   Maarten Lankhorst   fence: dma-buf cr...
340
  		ret = -ENOENT;
c701317a3   Daniel Vetter   dma-fence: Make -...
341
  	else if (!was_set && fence->ops->enable_signaling) {
f54d18670   Chris Wilson   dma-buf: Rename s...
342
  		trace_dma_fence_enable_signal(fence);
e941759c7   Maarten Lankhorst   fence: dma-buf cr...
343
344
  
  		if (!fence->ops->enable_signaling(fence)) {
f54d18670   Chris Wilson   dma-buf: Rename s...
345
  			dma_fence_signal_locked(fence);
e941759c7   Maarten Lankhorst   fence: dma-buf cr...
346
347
348
349
350
351
352
353
354
355
356
357
358
  			ret = -ENOENT;
  		}
  	}
  
  	if (!ret) {
  		cb->func = func;
  		list_add_tail(&cb->node, &fence->cb_list);
  	} else
  		INIT_LIST_HEAD(&cb->node);
  	spin_unlock_irqrestore(fence->lock, flags);
  
  	return ret;
  }
f54d18670   Chris Wilson   dma-buf: Rename s...
359
  EXPORT_SYMBOL(dma_fence_add_callback);
e941759c7   Maarten Lankhorst   fence: dma-buf cr...
360
361
  
  /**
d6c99f4bf   Chris Wilson   dma-fence: Wrap q...
362
   * dma_fence_get_status - returns the status upon completion
4dd3cdb28   Daniel Vetter   dma-fence: Polish...
363
   * @fence: the dma_fence to query
d6c99f4bf   Chris Wilson   dma-fence: Wrap q...
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
   *
   * This wraps dma_fence_get_status_locked() to return the error status
   * condition on a signaled fence. See dma_fence_get_status_locked() for more
   * details.
   *
   * Returns 0 if the fence has not yet been signaled, 1 if the fence has
   * been signaled without an error condition, or a negative error code
   * if the fence has been completed in err.
   */
  int dma_fence_get_status(struct dma_fence *fence)
  {
  	unsigned long flags;
  	int status;
  
  	spin_lock_irqsave(fence->lock, flags);
  	status = dma_fence_get_status_locked(fence);
  	spin_unlock_irqrestore(fence->lock, flags);
  
  	return status;
  }
  EXPORT_SYMBOL(dma_fence_get_status);
  
  /**
f54d18670   Chris Wilson   dma-buf: Rename s...
387
   * dma_fence_remove_callback - remove a callback from the signaling list
4dd3cdb28   Daniel Vetter   dma-fence: Polish...
388
389
   * @fence: the fence to wait on
   * @cb: the callback to remove
e941759c7   Maarten Lankhorst   fence: dma-buf cr...
390
391
   *
   * Remove a previously queued callback from the fence. This function returns
f353d71f7   Masanari Iida   treewide: Fix typ...
392
   * true if the callback is successfully removed, or false if the fence has
e941759c7   Maarten Lankhorst   fence: dma-buf cr...
393
394
395
396
397
398
399
   * already been signaled.
   *
   * *WARNING*:
   * Cancelling a callback should only be done if you really know what you're
   * doing, since deadlocks and race conditions could occur all too easily. For
   * this reason, it should only ever be done on hardware lockup recovery,
   * with a reference held to the fence.
4dd3cdb28   Daniel Vetter   dma-fence: Polish...
400
401
402
   *
   * Behaviour is undefined if @cb has not been added to @fence using
   * dma_fence_add_callback() beforehand.
e941759c7   Maarten Lankhorst   fence: dma-buf cr...
403
404
   */
  bool
f54d18670   Chris Wilson   dma-buf: Rename s...
405
  dma_fence_remove_callback(struct dma_fence *fence, struct dma_fence_cb *cb)
e941759c7   Maarten Lankhorst   fence: dma-buf cr...
406
407
408
409
410
411
412
413
414
415
416
417
418
419
  {
  	unsigned long flags;
  	bool ret;
  
  	spin_lock_irqsave(fence->lock, flags);
  
  	ret = !list_empty(&cb->node);
  	if (ret)
  		list_del_init(&cb->node);
  
  	spin_unlock_irqrestore(fence->lock, flags);
  
  	return ret;
  }
f54d18670   Chris Wilson   dma-buf: Rename s...
420
  EXPORT_SYMBOL(dma_fence_remove_callback);
e941759c7   Maarten Lankhorst   fence: dma-buf cr...
421
422
  
  struct default_wait_cb {
f54d18670   Chris Wilson   dma-buf: Rename s...
423
  	struct dma_fence_cb base;
e941759c7   Maarten Lankhorst   fence: dma-buf cr...
424
425
426
427
  	struct task_struct *task;
  };
  
  static void
f54d18670   Chris Wilson   dma-buf: Rename s...
428
  dma_fence_default_wait_cb(struct dma_fence *fence, struct dma_fence_cb *cb)
e941759c7   Maarten Lankhorst   fence: dma-buf cr...
429
430
431
432
433
434
435
436
  {
  	struct default_wait_cb *wait =
  		container_of(cb, struct default_wait_cb, base);
  
  	wake_up_state(wait->task, TASK_NORMAL);
  }
  
  /**
f54d18670   Chris Wilson   dma-buf: Rename s...
437
   * dma_fence_default_wait - default sleep until the fence gets signaled
e941759c7   Maarten Lankhorst   fence: dma-buf cr...
438
   * or until timeout elapses
4dd3cdb28   Daniel Vetter   dma-fence: Polish...
439
440
441
   * @fence: the fence to wait on
   * @intr: if true, do an interruptible wait
   * @timeout: timeout value in jiffies, or MAX_SCHEDULE_TIMEOUT
e941759c7   Maarten Lankhorst   fence: dma-buf cr...
442
443
   *
   * Returns -ERESTARTSYS if interrupted, 0 if the wait timed out, or the
bcc004b62   Alex Deucher   dma-buf/fence: ma...
444
445
446
   * remaining timeout in jiffies on success. If timeout is zero the value one is
   * returned if the fence is already signaled for consistency with other
   * functions taking a jiffies timeout.
e941759c7   Maarten Lankhorst   fence: dma-buf cr...
447
448
   */
  signed long
f54d18670   Chris Wilson   dma-buf: Rename s...
449
  dma_fence_default_wait(struct dma_fence *fence, bool intr, signed long timeout)
e941759c7   Maarten Lankhorst   fence: dma-buf cr...
450
451
452
  {
  	struct default_wait_cb cb;
  	unsigned long flags;
bcc004b62   Alex Deucher   dma-buf/fence: ma...
453
  	signed long ret = timeout ? timeout : 1;
e941759c7   Maarten Lankhorst   fence: dma-buf cr...
454
  	bool was_set;
f54d18670   Chris Wilson   dma-buf: Rename s...
455
  	if (test_bit(DMA_FENCE_FLAG_SIGNALED_BIT, &fence->flags))
bcc004b62   Alex Deucher   dma-buf/fence: ma...
456
  		return ret;
e941759c7   Maarten Lankhorst   fence: dma-buf cr...
457
458
459
460
461
462
463
  
  	spin_lock_irqsave(fence->lock, flags);
  
  	if (intr && signal_pending(current)) {
  		ret = -ERESTARTSYS;
  		goto out;
  	}
f54d18670   Chris Wilson   dma-buf: Rename s...
464
465
  	was_set = test_and_set_bit(DMA_FENCE_FLAG_ENABLE_SIGNAL_BIT,
  				   &fence->flags);
e941759c7   Maarten Lankhorst   fence: dma-buf cr...
466

f54d18670   Chris Wilson   dma-buf: Rename s...
467
  	if (test_bit(DMA_FENCE_FLAG_SIGNALED_BIT, &fence->flags))
e941759c7   Maarten Lankhorst   fence: dma-buf cr...
468
  		goto out;
c701317a3   Daniel Vetter   dma-fence: Make -...
469
  	if (!was_set && fence->ops->enable_signaling) {
f54d18670   Chris Wilson   dma-buf: Rename s...
470
  		trace_dma_fence_enable_signal(fence);
e941759c7   Maarten Lankhorst   fence: dma-buf cr...
471
472
  
  		if (!fence->ops->enable_signaling(fence)) {
f54d18670   Chris Wilson   dma-buf: Rename s...
473
  			dma_fence_signal_locked(fence);
e941759c7   Maarten Lankhorst   fence: dma-buf cr...
474
475
476
  			goto out;
  		}
  	}
03c0c5f66   Andres Rodriguez   dma-buf: avoid sc...
477
478
479
480
  	if (!timeout) {
  		ret = 0;
  		goto out;
  	}
f54d18670   Chris Wilson   dma-buf: Rename s...
481
  	cb.base.func = dma_fence_default_wait_cb;
e941759c7   Maarten Lankhorst   fence: dma-buf cr...
482
483
  	cb.task = current;
  	list_add(&cb.base.node, &fence->cb_list);
f54d18670   Chris Wilson   dma-buf: Rename s...
484
  	while (!test_bit(DMA_FENCE_FLAG_SIGNALED_BIT, &fence->flags) && ret > 0) {
e941759c7   Maarten Lankhorst   fence: dma-buf cr...
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
  		if (intr)
  			__set_current_state(TASK_INTERRUPTIBLE);
  		else
  			__set_current_state(TASK_UNINTERRUPTIBLE);
  		spin_unlock_irqrestore(fence->lock, flags);
  
  		ret = schedule_timeout(ret);
  
  		spin_lock_irqsave(fence->lock, flags);
  		if (ret > 0 && intr && signal_pending(current))
  			ret = -ERESTARTSYS;
  	}
  
  	if (!list_empty(&cb.base.node))
  		list_del(&cb.base.node);
  	__set_current_state(TASK_RUNNING);
  
  out:
  	spin_unlock_irqrestore(fence->lock, flags);
  	return ret;
  }
f54d18670   Chris Wilson   dma-buf: Rename s...
506
  EXPORT_SYMBOL(dma_fence_default_wait);
e941759c7   Maarten Lankhorst   fence: dma-buf cr...
507

a519435a9   Christian König   dma-buf/fence: ad...
508
  static bool
7392b4bb7   monk.liu   dma-buf: return i...
509
510
  dma_fence_test_signaled_any(struct dma_fence **fences, uint32_t count,
  			    uint32_t *idx)
a519435a9   Christian König   dma-buf/fence: ad...
511
512
513
514
  {
  	int i;
  
  	for (i = 0; i < count; ++i) {
f54d18670   Chris Wilson   dma-buf: Rename s...
515
  		struct dma_fence *fence = fences[i];
7392b4bb7   monk.liu   dma-buf: return i...
516
517
518
  		if (test_bit(DMA_FENCE_FLAG_SIGNALED_BIT, &fence->flags)) {
  			if (idx)
  				*idx = i;
a519435a9   Christian König   dma-buf/fence: ad...
519
  			return true;
7392b4bb7   monk.liu   dma-buf: return i...
520
  		}
a519435a9   Christian König   dma-buf/fence: ad...
521
522
523
524
525
  	}
  	return false;
  }
  
  /**
f54d18670   Chris Wilson   dma-buf: Rename s...
526
   * dma_fence_wait_any_timeout - sleep until any fence gets signaled
a519435a9   Christian König   dma-buf/fence: ad...
527
   * or until timeout elapses
4dd3cdb28   Daniel Vetter   dma-fence: Polish...
528
529
530
531
532
533
   * @fences: array of fences to wait on
   * @count: number of fences to wait on
   * @intr: if true, do an interruptible wait
   * @timeout: timeout value in jiffies, or MAX_SCHEDULE_TIMEOUT
   * @idx: used to store the first signaled fence index, meaningful only on
   *	positive return
a519435a9   Christian König   dma-buf/fence: ad...
534
535
536
537
538
539
540
541
   *
   * Returns -EINVAL on custom fence wait implementation, -ERESTARTSYS if
   * interrupted, 0 if the wait timed out, or the remaining timeout in jiffies
   * on success.
   *
   * Synchronous waits for the first fence in the array to be signaled. The
   * caller needs to hold a reference to all fences in the array, otherwise a
   * fence might be freed before return, resulting in undefined behavior.
4dd3cdb28   Daniel Vetter   dma-fence: Polish...
542
543
   *
   * See also dma_fence_wait() and dma_fence_wait_timeout().
a519435a9   Christian König   dma-buf/fence: ad...
544
545
   */
  signed long
f54d18670   Chris Wilson   dma-buf: Rename s...
546
  dma_fence_wait_any_timeout(struct dma_fence **fences, uint32_t count,
7392b4bb7   monk.liu   dma-buf: return i...
547
  			   bool intr, signed long timeout, uint32_t *idx)
a519435a9   Christian König   dma-buf/fence: ad...
548
549
550
551
552
553
554
555
556
557
  {
  	struct default_wait_cb *cb;
  	signed long ret = timeout;
  	unsigned i;
  
  	if (WARN_ON(!fences || !count || timeout < 0))
  		return -EINVAL;
  
  	if (timeout == 0) {
  		for (i = 0; i < count; ++i)
7392b4bb7   monk.liu   dma-buf: return i...
558
559
560
  			if (dma_fence_is_signaled(fences[i])) {
  				if (idx)
  					*idx = i;
a519435a9   Christian König   dma-buf/fence: ad...
561
  				return 1;
7392b4bb7   monk.liu   dma-buf: return i...
562
  			}
a519435a9   Christian König   dma-buf/fence: ad...
563
564
565
566
567
568
569
570
571
572
573
  
  		return 0;
  	}
  
  	cb = kcalloc(count, sizeof(struct default_wait_cb), GFP_KERNEL);
  	if (cb == NULL) {
  		ret = -ENOMEM;
  		goto err_free_cb;
  	}
  
  	for (i = 0; i < count; ++i) {
f54d18670   Chris Wilson   dma-buf: Rename s...
574
  		struct dma_fence *fence = fences[i];
a519435a9   Christian König   dma-buf/fence: ad...
575

a519435a9   Christian König   dma-buf/fence: ad...
576
  		cb[i].task = current;
f54d18670   Chris Wilson   dma-buf: Rename s...
577
578
  		if (dma_fence_add_callback(fence, &cb[i].base,
  					   dma_fence_default_wait_cb)) {
a519435a9   Christian König   dma-buf/fence: ad...
579
  			/* This fence is already signaled */
7392b4bb7   monk.liu   dma-buf: return i...
580
581
  			if (idx)
  				*idx = i;
a519435a9   Christian König   dma-buf/fence: ad...
582
583
584
585
586
587
588
589
590
  			goto fence_rm_cb;
  		}
  	}
  
  	while (ret > 0) {
  		if (intr)
  			set_current_state(TASK_INTERRUPTIBLE);
  		else
  			set_current_state(TASK_UNINTERRUPTIBLE);
7392b4bb7   monk.liu   dma-buf: return i...
591
  		if (dma_fence_test_signaled_any(fences, count, idx))
a519435a9   Christian König   dma-buf/fence: ad...
592
593
594
595
596
597
598
599
600
601
602
603
  			break;
  
  		ret = schedule_timeout(ret);
  
  		if (ret > 0 && intr && signal_pending(current))
  			ret = -ERESTARTSYS;
  	}
  
  	__set_current_state(TASK_RUNNING);
  
  fence_rm_cb:
  	while (i-- > 0)
f54d18670   Chris Wilson   dma-buf: Rename s...
604
  		dma_fence_remove_callback(fences[i], &cb[i].base);
a519435a9   Christian König   dma-buf/fence: ad...
605
606
607
608
609
610
  
  err_free_cb:
  	kfree(cb);
  
  	return ret;
  }
f54d18670   Chris Wilson   dma-buf: Rename s...
611
  EXPORT_SYMBOL(dma_fence_wait_any_timeout);
a519435a9   Christian König   dma-buf/fence: ad...
612

e941759c7   Maarten Lankhorst   fence: dma-buf cr...
613
  /**
f54d18670   Chris Wilson   dma-buf: Rename s...
614
   * dma_fence_init - Initialize a custom fence.
4dd3cdb28   Daniel Vetter   dma-fence: Polish...
615
616
617
618
619
   * @fence: the fence to initialize
   * @ops: the dma_fence_ops for operations on this fence
   * @lock: the irqsafe spinlock to use for locking this fence
   * @context: the execution context this fence is run on
   * @seqno: a linear increasing sequence number for this context
e941759c7   Maarten Lankhorst   fence: dma-buf cr...
620
621
622
   *
   * Initializes an allocated fence, the caller doesn't have to keep its
   * refcount after committing with this fence, but it will need to hold a
4dd3cdb28   Daniel Vetter   dma-fence: Polish...
623
   * refcount again if &dma_fence_ops.enable_signaling gets called.
e941759c7   Maarten Lankhorst   fence: dma-buf cr...
624
625
   *
   * context and seqno are used for easy comparison between fences, allowing
4dd3cdb28   Daniel Vetter   dma-fence: Polish...
626
   * to check which fence is later by simply using dma_fence_later().
e941759c7   Maarten Lankhorst   fence: dma-buf cr...
627
628
   */
  void
f54d18670   Chris Wilson   dma-buf: Rename s...
629
  dma_fence_init(struct dma_fence *fence, const struct dma_fence_ops *ops,
b312d8ca3   Christian König   dma-buf: make fen...
630
  	       spinlock_t *lock, u64 context, u64 seqno)
e941759c7   Maarten Lankhorst   fence: dma-buf cr...
631
632
  {
  	BUG_ON(!lock);
418cc6ca0   Daniel Vetter   dma-fence: Make -...
633
  	BUG_ON(!ops || !ops->get_driver_name || !ops->get_timeline_name);
e941759c7   Maarten Lankhorst   fence: dma-buf cr...
634
635
636
637
638
639
640
641
  
  	kref_init(&fence->refcount);
  	fence->ops = ops;
  	INIT_LIST_HEAD(&fence->cb_list);
  	fence->lock = lock;
  	fence->context = context;
  	fence->seqno = seqno;
  	fence->flags = 0UL;
a009e975d   Chris Wilson   dma-fence: Introd...
642
  	fence->error = 0;
e941759c7   Maarten Lankhorst   fence: dma-buf cr...
643

f54d18670   Chris Wilson   dma-buf: Rename s...
644
  	trace_dma_fence_init(fence);
e941759c7   Maarten Lankhorst   fence: dma-buf cr...
645
  }
f54d18670   Chris Wilson   dma-buf: Rename s...
646
  EXPORT_SYMBOL(dma_fence_init);