| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150 |
- /* SPDX-License-Identifier: GPL-2.0 OR MIT */
- #ifndef __DRM_EXEC_H__
- #define __DRM_EXEC_H__
- #include <linux/compiler.h>
- #include <linux/ww_mutex.h>
- #define DRM_EXEC_INTERRUPTIBLE_WAIT BIT(0)
- #define DRM_EXEC_IGNORE_DUPLICATES BIT(1)
- struct drm_gem_object;
- /**
- * struct drm_exec - Execution context
- */
- struct drm_exec {
- /**
- * @flags: Flags to control locking behavior
- */
- u32 flags;
- /**
- * @ticket: WW ticket used for acquiring locks
- */
- struct ww_acquire_ctx ticket;
- /**
- * @num_objects: number of objects locked
- */
- unsigned int num_objects;
- /**
- * @max_objects: maximum objects in array
- */
- unsigned int max_objects;
- /**
- * @objects: array of the locked objects
- */
- struct drm_gem_object **objects;
- /**
- * @contended: contended GEM object we backed off for
- */
- struct drm_gem_object *contended;
- /**
- * @prelocked: already locked GEM object due to contention
- */
- struct drm_gem_object *prelocked;
- };
- /**
- * drm_exec_obj() - Return the object for a give drm_exec index
- * @exec: Pointer to the drm_exec context
- * @index: The index.
- *
- * Return: Pointer to the locked object corresponding to @index if
- * index is within the number of locked objects. NULL otherwise.
- */
- static inline struct drm_gem_object *
- drm_exec_obj(struct drm_exec *exec, unsigned long index)
- {
- return index < exec->num_objects ? exec->objects[index] : NULL;
- }
- /**
- * drm_exec_for_each_locked_object - iterate over all the locked objects
- * @exec: drm_exec object
- * @index: unsigned long index for the iteration
- * @obj: the current GEM object
- *
- * Iterate over all the locked GEM objects inside the drm_exec object.
- */
- #define drm_exec_for_each_locked_object(exec, index, obj) \
- for ((index) = 0; ((obj) = drm_exec_obj(exec, index)); ++(index))
- /**
- * drm_exec_for_each_locked_object_reverse - iterate over all the locked
- * objects in reverse locking order
- * @exec: drm_exec object
- * @index: unsigned long index for the iteration
- * @obj: the current GEM object
- *
- * Iterate over all the locked GEM objects inside the drm_exec object in
- * reverse locking order. Note that @index may go below zero and wrap,
- * but that will be caught by drm_exec_obj(), returning a NULL object.
- */
- #define drm_exec_for_each_locked_object_reverse(exec, index, obj) \
- for ((index) = (exec)->num_objects - 1; \
- ((obj) = drm_exec_obj(exec, index)); --(index))
- /**
- * drm_exec_until_all_locked - loop until all GEM objects are locked
- * @exec: drm_exec object
- *
- * Core functionality of the drm_exec object. Loops until all GEM objects are
- * locked and no more contention exists. At the beginning of the loop it is
- * guaranteed that no GEM object is locked.
- *
- * Since labels can't be defined local to the loops body we use a jump pointer
- * to make sure that the retry is only used from within the loops body.
- */
- #define drm_exec_until_all_locked(exec) \
- __PASTE(__drm_exec_, __LINE__): \
- for (void *__drm_exec_retry_ptr; ({ \
- __drm_exec_retry_ptr = &&__PASTE(__drm_exec_, __LINE__);\
- (void)__drm_exec_retry_ptr; \
- drm_exec_cleanup(exec); \
- });)
- /**
- * drm_exec_retry_on_contention - restart the loop to grap all locks
- * @exec: drm_exec object
- *
- * Control flow helper to continue when a contention was detected and we need to
- * clean up and re-start the loop to prepare all GEM objects.
- */
- #define drm_exec_retry_on_contention(exec) \
- do { \
- if (unlikely(drm_exec_is_contended(exec))) \
- goto *__drm_exec_retry_ptr; \
- } while (0)
- /**
- * drm_exec_is_contended - check for contention
- * @exec: drm_exec object
- *
- * Returns true if the drm_exec object has run into some contention while
- * locking a GEM object and needs to clean up.
- */
- static inline bool drm_exec_is_contended(struct drm_exec *exec)
- {
- return !!exec->contended;
- }
- void drm_exec_init(struct drm_exec *exec, u32 flags, unsigned nr);
- void drm_exec_fini(struct drm_exec *exec);
- bool drm_exec_cleanup(struct drm_exec *exec);
- int drm_exec_lock_obj(struct drm_exec *exec, struct drm_gem_object *obj);
- void drm_exec_unlock_obj(struct drm_exec *exec, struct drm_gem_object *obj);
- int drm_exec_prepare_obj(struct drm_exec *exec, struct drm_gem_object *obj,
- unsigned int num_fences);
- int drm_exec_prepare_array(struct drm_exec *exec,
- struct drm_gem_object **objects,
- unsigned int num_objects,
- unsigned int num_fences);
- #endif
|