/* SPDX-License-Identifier: GPL-2.0 OR MIT */12#ifndef __DRM_EXEC_H__3#define __DRM_EXEC_H__45#include <linux/compiler.h>6#include <linux/ww_mutex.h>78#define DRM_EXEC_INTERRUPTIBLE_WAIT BIT(0)9#define DRM_EXEC_IGNORE_DUPLICATES BIT(1)1011struct drm_gem_object;1213/**14* struct drm_exec - Execution context15*/16struct drm_exec {17/**18* @flags: Flags to control locking behavior19*/20u32 flags;2122/**23* @ticket: WW ticket used for acquiring locks24*/25struct ww_acquire_ctx ticket;2627/**28* @num_objects: number of objects locked29*/30unsigned int num_objects;3132/**33* @max_objects: maximum objects in array34*/35unsigned int max_objects;3637/**38* @objects: array of the locked objects39*/40struct drm_gem_object **objects;4142/**43* @contended: contended GEM object we backed off for44*/45struct drm_gem_object *contended;4647/**48* @prelocked: already locked GEM object due to contention49*/50struct drm_gem_object *prelocked;51};5253/**54* drm_exec_obj() - Return the object for a give drm_exec index55* @exec: Pointer to the drm_exec context56* @index: The index.57*58* Return: Pointer to the locked object corresponding to @index if59* index is within the number of locked objects. NULL otherwise.60*/61static inline struct drm_gem_object *62drm_exec_obj(struct drm_exec *exec, unsigned long index)63{64return index < exec->num_objects ? exec->objects[index] : NULL;65}6667/**68* drm_exec_for_each_locked_object - iterate over all the locked objects69* @exec: drm_exec object70* @index: unsigned long index for the iteration71* @obj: the current GEM object72*73* Iterate over all the locked GEM objects inside the drm_exec object.74*/75#define drm_exec_for_each_locked_object(exec, index, obj) \76for ((index) = 0; ((obj) = drm_exec_obj(exec, index)); ++(index))7778/**79* drm_exec_for_each_locked_object_reverse - iterate over all the locked80* objects in reverse locking order81* @exec: drm_exec object82* @index: unsigned long index for the iteration83* @obj: the current GEM object84*85* Iterate over all the locked GEM objects inside the drm_exec object in86* reverse locking order. Note that @index may go below zero and wrap,87* but that will be caught by drm_exec_obj(), returning a NULL object.88*/89#define drm_exec_for_each_locked_object_reverse(exec, index, obj) \90for ((index) = (exec)->num_objects - 1; \91((obj) = drm_exec_obj(exec, index)); --(index))9293/**94* drm_exec_until_all_locked - loop until all GEM objects are locked95* @exec: drm_exec object96*97* Core functionality of the drm_exec object. Loops until all GEM objects are98* locked and no more contention exists. At the beginning of the loop it is99* guaranteed that no GEM object is locked.100*101* Since labels can't be defined local to the loops body we use a jump pointer102* to make sure that the retry is only used from within the loops body.103*/104#define drm_exec_until_all_locked(exec) \105__PASTE(__drm_exec_, __LINE__): \106for (void *__drm_exec_retry_ptr; ({ \107__drm_exec_retry_ptr = &&__PASTE(__drm_exec_, __LINE__);\108(void)__drm_exec_retry_ptr; \109drm_exec_cleanup(exec); \110});)111112/**113* drm_exec_retry_on_contention - restart the loop to grap all locks114* @exec: drm_exec object115*116* Control flow helper to continue when a contention was detected and we need to117* clean up and re-start the loop to prepare all GEM objects.118*/119#define drm_exec_retry_on_contention(exec) \120do { \121if (unlikely(drm_exec_is_contended(exec))) \122goto *__drm_exec_retry_ptr; \123} while (0)124125/**126* drm_exec_is_contended - check for contention127* @exec: drm_exec object128*129* Returns true if the drm_exec object has run into some contention while130* locking a GEM object and needs to clean up.131*/132static inline bool drm_exec_is_contended(struct drm_exec *exec)133{134return !!exec->contended;135}136137void drm_exec_init(struct drm_exec *exec, u32 flags, unsigned nr);138void drm_exec_fini(struct drm_exec *exec);139bool drm_exec_cleanup(struct drm_exec *exec);140int drm_exec_lock_obj(struct drm_exec *exec, struct drm_gem_object *obj);141void drm_exec_unlock_obj(struct drm_exec *exec, struct drm_gem_object *obj);142int drm_exec_prepare_obj(struct drm_exec *exec, struct drm_gem_object *obj,143unsigned int num_fences);144int drm_exec_prepare_array(struct drm_exec *exec,145struct drm_gem_object **objects,146unsigned int num_objects,147unsigned int num_fences);148149#endif150151152