1  /* SPDX-License-Identifier: GPL-2.0 OR MIT */
2  
3  #ifndef __DRM_EXEC_H__
4  #define __DRM_EXEC_H__
5  
6  #include <linux/compiler.h>
7  #include <linux/ww_mutex.h>
8  
9  #define DRM_EXEC_INTERRUPTIBLE_WAIT	BIT(0)
10  #define DRM_EXEC_IGNORE_DUPLICATES	BIT(1)
11  
12  struct drm_gem_object;
13  
14  /**
15   * struct drm_exec - Execution context
16   */
17  struct drm_exec {
18  	/**
19  	 * @flags: Flags to control locking behavior
20  	 */
21  	u32                     flags;
22  
23  	/**
24  	 * @ticket: WW ticket used for acquiring locks
25  	 */
26  	struct ww_acquire_ctx	ticket;
27  
28  	/**
29  	 * @num_objects: number of objects locked
30  	 */
31  	unsigned int		num_objects;
32  
33  	/**
34  	 * @max_objects: maximum objects in array
35  	 */
36  	unsigned int		max_objects;
37  
38  	/**
39  	 * @objects: array of the locked objects
40  	 */
41  	struct drm_gem_object	**objects;
42  
43  	/**
44  	 * @contended: contended GEM object we backed off for
45  	 */
46  	struct drm_gem_object	*contended;
47  
48  	/**
49  	 * @prelocked: already locked GEM object due to contention
50  	 */
51  	struct drm_gem_object *prelocked;
52  };
53  
54  /**
55   * drm_exec_obj() - Return the object for a give drm_exec index
56   * @exec: Pointer to the drm_exec context
57   * @index: The index.
58   *
59   * Return: Pointer to the locked object corresponding to @index if
60   * index is within the number of locked objects. NULL otherwise.
61   */
62  static inline struct drm_gem_object *
drm_exec_obj(struct drm_exec * exec,unsigned long index)63  drm_exec_obj(struct drm_exec *exec, unsigned long index)
64  {
65  	return index < exec->num_objects ? exec->objects[index] : NULL;
66  }
67  
68  /**
69   * drm_exec_for_each_locked_object - iterate over all the locked objects
70   * @exec: drm_exec object
71   * @index: unsigned long index for the iteration
72   * @obj: the current GEM object
73   *
74   * Iterate over all the locked GEM objects inside the drm_exec object.
75   */
76  #define drm_exec_for_each_locked_object(exec, index, obj)		\
77  	for ((index) = 0; ((obj) = drm_exec_obj(exec, index)); ++(index))
78  
79  /**
80   * drm_exec_for_each_locked_object_reverse - iterate over all the locked
81   * objects in reverse locking order
82   * @exec: drm_exec object
83   * @index: unsigned long index for the iteration
84   * @obj: the current GEM object
85   *
86   * Iterate over all the locked GEM objects inside the drm_exec object in
87   * reverse locking order. Note that @index may go below zero and wrap,
88   * but that will be caught by drm_exec_obj(), returning a NULL object.
89   */
90  #define drm_exec_for_each_locked_object_reverse(exec, index, obj)	\
91  	for ((index) = (exec)->num_objects - 1;				\
92  	     ((obj) = drm_exec_obj(exec, index)); --(index))
93  
94  /**
95   * drm_exec_until_all_locked - loop until all GEM objects are locked
96   * @exec: drm_exec object
97   *
98   * Core functionality of the drm_exec object. Loops until all GEM objects are
99   * locked and no more contention exists. At the beginning of the loop it is
100   * guaranteed that no GEM object is locked.
101   *
102   * Since labels can't be defined local to the loops body we use a jump pointer
103   * to make sure that the retry is only used from within the loops body.
104   */
105  #define drm_exec_until_all_locked(exec)					\
106  __PASTE(__drm_exec_, __LINE__):						\
107  	for (void *__drm_exec_retry_ptr; ({				\
108  		__drm_exec_retry_ptr = &&__PASTE(__drm_exec_, __LINE__);\
109  		(void)__drm_exec_retry_ptr;				\
110  		drm_exec_cleanup(exec);					\
111  	});)
112  
113  /**
114   * drm_exec_retry_on_contention - restart the loop to grap all locks
115   * @exec: drm_exec object
116   *
117   * Control flow helper to continue when a contention was detected and we need to
118   * clean up and re-start the loop to prepare all GEM objects.
119   */
120  #define drm_exec_retry_on_contention(exec)			\
121  	do {							\
122  		if (unlikely(drm_exec_is_contended(exec)))	\
123  			goto *__drm_exec_retry_ptr;		\
124  	} while (0)
125  
126  /**
127   * drm_exec_is_contended - check for contention
128   * @exec: drm_exec object
129   *
130   * Returns true if the drm_exec object has run into some contention while
131   * locking a GEM object and needs to clean up.
132   */
drm_exec_is_contended(struct drm_exec * exec)133  static inline bool drm_exec_is_contended(struct drm_exec *exec)
134  {
135  	return !!exec->contended;
136  }
137  
138  void drm_exec_init(struct drm_exec *exec, u32 flags, unsigned nr);
139  void drm_exec_fini(struct drm_exec *exec);
140  bool drm_exec_cleanup(struct drm_exec *exec);
141  int drm_exec_lock_obj(struct drm_exec *exec, struct drm_gem_object *obj);
142  void drm_exec_unlock_obj(struct drm_exec *exec, struct drm_gem_object *obj);
143  int drm_exec_prepare_obj(struct drm_exec *exec, struct drm_gem_object *obj,
144  			 unsigned int num_fences);
145  int drm_exec_prepare_array(struct drm_exec *exec,
146  			   struct drm_gem_object **objects,
147  			   unsigned int num_objects,
148  			   unsigned int num_fences);
149  
150  #endif
151