Merge tag 'trace-printf-v6.13' of git://git.kernel.org/pub/scm/linux/kernel/git/trace...
[drm/drm-misc.git] / include / media / media-request.h
blobd4ac557678a78372222704400c8c96cf3150b9d9
1 // SPDX-License-Identifier: GPL-2.0
2 /*
3 * Media device request objects
5 * Copyright 2018 Cisco Systems, Inc. and/or its affiliates. All rights reserved.
6 * Copyright (C) 2018 Intel Corporation
8 * Author: Hans Verkuil <hansverk@cisco.com>
9 * Author: Sakari Ailus <sakari.ailus@linux.intel.com>
12 #ifndef MEDIA_REQUEST_H
13 #define MEDIA_REQUEST_H
15 #include <linux/list.h>
16 #include <linux/slab.h>
17 #include <linux/spinlock.h>
18 #include <linux/refcount.h>
20 #include <media/media-device.h>
22 /**
23 * enum media_request_state - media request state
25 * @MEDIA_REQUEST_STATE_IDLE: Idle
26 * @MEDIA_REQUEST_STATE_VALIDATING: Validating the request, no state changes
27 * allowed
28 * @MEDIA_REQUEST_STATE_QUEUED: Queued
29 * @MEDIA_REQUEST_STATE_COMPLETE: Completed, the request is done
30 * @MEDIA_REQUEST_STATE_CLEANING: Cleaning, the request is being re-inited
31 * @MEDIA_REQUEST_STATE_UPDATING: The request is being updated, i.e.
32 * request objects are being added,
33 * modified or removed
34 * @NR_OF_MEDIA_REQUEST_STATE: The number of media request states, used
35 * internally for sanity check purposes
37 enum media_request_state {
38 MEDIA_REQUEST_STATE_IDLE,
39 MEDIA_REQUEST_STATE_VALIDATING,
40 MEDIA_REQUEST_STATE_QUEUED,
41 MEDIA_REQUEST_STATE_COMPLETE,
42 MEDIA_REQUEST_STATE_CLEANING,
43 MEDIA_REQUEST_STATE_UPDATING,
44 NR_OF_MEDIA_REQUEST_STATE,
47 struct media_request_object;
49 /**
50 * struct media_request - Media device request
51 * @mdev: Media device this request belongs to
52 * @kref: Reference count
53 * @debug_str: Prefix for debug messages (process name:fd)
54 * @state: The state of the request
55 * @updating_count: count the number of request updates that are in progress
56 * @access_count: count the number of request accesses that are in progress
57 * @objects: List of @struct media_request_object request objects
58 * @num_incomplete_objects: The number of incomplete objects in the request
59 * @poll_wait: Wait queue for poll
60 * @lock: Serializes access to this struct
62 struct media_request {
63 struct media_device *mdev;
64 struct kref kref;
65 char debug_str[TASK_COMM_LEN + 11];
66 enum media_request_state state;
67 unsigned int updating_count;
68 unsigned int access_count;
69 struct list_head objects;
70 unsigned int num_incomplete_objects;
71 wait_queue_head_t poll_wait;
72 spinlock_t lock;
75 #ifdef CONFIG_MEDIA_CONTROLLER
77 /**
78 * media_request_lock_for_access - Lock the request to access its objects
80 * @req: The media request
82 * Use before accessing a completed request. A reference to the request must
83 * be held during the access. This usually takes place automatically through
84 * a file handle. Use @media_request_unlock_for_access when done.
86 static inline int __must_check
87 media_request_lock_for_access(struct media_request *req)
89 unsigned long flags;
90 int ret = -EBUSY;
92 spin_lock_irqsave(&req->lock, flags);
93 if (req->state == MEDIA_REQUEST_STATE_COMPLETE) {
94 req->access_count++;
95 ret = 0;
97 spin_unlock_irqrestore(&req->lock, flags);
99 return ret;
103 * media_request_unlock_for_access - Unlock a request previously locked for
104 * access
106 * @req: The media request
108 * Unlock a request that has previously been locked using
109 * @media_request_lock_for_access.
111 static inline void media_request_unlock_for_access(struct media_request *req)
113 unsigned long flags;
115 spin_lock_irqsave(&req->lock, flags);
116 if (!WARN_ON(!req->access_count))
117 req->access_count--;
118 spin_unlock_irqrestore(&req->lock, flags);
122 * media_request_lock_for_update - Lock the request for updating its objects
124 * @req: The media request
126 * Use before updating a request, i.e. adding, modifying or removing a request
127 * object in it. A reference to the request must be held during the update. This
128 * usually takes place automatically through a file handle. Use
129 * @media_request_unlock_for_update when done.
131 static inline int __must_check
132 media_request_lock_for_update(struct media_request *req)
134 unsigned long flags;
135 int ret = 0;
137 spin_lock_irqsave(&req->lock, flags);
138 if (req->state == MEDIA_REQUEST_STATE_IDLE ||
139 req->state == MEDIA_REQUEST_STATE_UPDATING) {
140 req->state = MEDIA_REQUEST_STATE_UPDATING;
141 req->updating_count++;
142 } else {
143 ret = -EBUSY;
145 spin_unlock_irqrestore(&req->lock, flags);
147 return ret;
151 * media_request_unlock_for_update - Unlock a request previously locked for
152 * update
154 * @req: The media request
156 * Unlock a request that has previously been locked using
157 * @media_request_lock_for_update.
159 static inline void media_request_unlock_for_update(struct media_request *req)
161 unsigned long flags;
163 spin_lock_irqsave(&req->lock, flags);
164 WARN_ON(req->updating_count <= 0);
165 if (!--req->updating_count)
166 req->state = MEDIA_REQUEST_STATE_IDLE;
167 spin_unlock_irqrestore(&req->lock, flags);
171 * media_request_get - Get the media request
173 * @req: The media request
175 * Get the media request.
177 static inline void media_request_get(struct media_request *req)
179 kref_get(&req->kref);
183 * media_request_put - Put the media request
185 * @req: The media request
187 * Put the media request. The media request will be released
188 * when the refcount reaches 0.
190 void media_request_put(struct media_request *req);
193 * media_request_get_by_fd - Get a media request by fd
195 * @mdev: Media device this request belongs to
196 * @request_fd: The file descriptor of the request
198 * Get the request represented by @request_fd that is owned
199 * by the media device.
201 * Return a -EBADR error pointer if requests are not supported
202 * by this driver. Return -EINVAL if the request was not found.
203 * Return the pointer to the request if found: the caller will
204 * have to call @media_request_put when it finished using the
205 * request.
207 struct media_request *
208 media_request_get_by_fd(struct media_device *mdev, int request_fd);
211 * media_request_alloc - Allocate the media request
213 * @mdev: Media device this request belongs to
214 * @alloc_fd: Store the request's file descriptor in this int
216 * Allocated the media request and put the fd in @alloc_fd.
218 int media_request_alloc(struct media_device *mdev,
219 int *alloc_fd);
221 #else
223 static inline void media_request_get(struct media_request *req)
227 static inline void media_request_put(struct media_request *req)
231 static inline struct media_request *
232 media_request_get_by_fd(struct media_device *mdev, int request_fd)
234 return ERR_PTR(-EBADR);
237 #endif
240 * struct media_request_object_ops - Media request object operations
241 * @prepare: Validate and prepare the request object, optional.
242 * @unprepare: Unprepare the request object, optional.
243 * @queue: Queue the request object, optional.
244 * @unbind: Unbind the request object, optional.
245 * @release: Release the request object, required.
247 struct media_request_object_ops {
248 int (*prepare)(struct media_request_object *object);
249 void (*unprepare)(struct media_request_object *object);
250 void (*queue)(struct media_request_object *object);
251 void (*unbind)(struct media_request_object *object);
252 void (*release)(struct media_request_object *object);
256 * struct media_request_object - An opaque object that belongs to a media
257 * request
259 * @ops: object's operations
260 * @priv: object's priv pointer
261 * @req: the request this object belongs to (can be NULL)
262 * @list: List entry of the object for @struct media_request
263 * @kref: Reference count of the object, acquire before releasing req->lock
264 * @completed: If true, then this object was completed.
266 * An object related to the request. This struct is always embedded in
267 * another struct that contains the actual data for this request object.
269 struct media_request_object {
270 const struct media_request_object_ops *ops;
271 void *priv;
272 struct media_request *req;
273 struct list_head list;
274 struct kref kref;
275 bool completed;
278 #ifdef CONFIG_MEDIA_CONTROLLER
281 * media_request_object_get - Get a media request object
283 * @obj: The object
285 * Get a media request object.
287 static inline void media_request_object_get(struct media_request_object *obj)
289 kref_get(&obj->kref);
293 * media_request_object_put - Put a media request object
295 * @obj: The object
297 * Put a media request object. Once all references are gone, the
298 * object's memory is released.
300 void media_request_object_put(struct media_request_object *obj);
303 * media_request_object_find - Find an object in a request
305 * @req: The media request
306 * @ops: Find an object with this ops value
307 * @priv: Find an object with this priv value
309 * Both @ops and @priv must be non-NULL.
311 * Returns the object pointer or NULL if not found. The caller must
312 * call media_request_object_put() once it finished using the object.
314 * Since this function needs to walk the list of objects it takes
315 * the @req->lock spin lock to make this safe.
317 struct media_request_object *
318 media_request_object_find(struct media_request *req,
319 const struct media_request_object_ops *ops,
320 void *priv);
323 * media_request_object_init - Initialise a media request object
325 * @obj: The object
327 * Initialise a media request object. The object will be released using the
328 * release callback of the ops once it has no references (this function
329 * initialises references to one).
331 void media_request_object_init(struct media_request_object *obj);
334 * media_request_object_bind - Bind a media request object to a request
336 * @req: The media request
337 * @ops: The object ops for this object
338 * @priv: A driver-specific priv pointer associated with this object
339 * @is_buffer: Set to true if the object a buffer object.
340 * @obj: The object
342 * Bind this object to the request and set the ops and priv values of
343 * the object so it can be found later with media_request_object_find().
345 * Every bound object must be unbound or completed by the kernel at some
346 * point in time, otherwise the request will never complete. When the
347 * request is released all completed objects will be unbound by the
348 * request core code.
350 * Buffer objects will be added to the end of the request's object
351 * list, non-buffer objects will be added to the front of the list.
352 * This ensures that all buffer objects are at the end of the list
353 * and that all non-buffer objects that they depend on are processed
354 * first.
356 int media_request_object_bind(struct media_request *req,
357 const struct media_request_object_ops *ops,
358 void *priv, bool is_buffer,
359 struct media_request_object *obj);
362 * media_request_object_unbind - Unbind a media request object
364 * @obj: The object
366 * Unbind the media request object from the request.
368 void media_request_object_unbind(struct media_request_object *obj);
371 * media_request_object_complete - Mark the media request object as complete
373 * @obj: The object
375 * Mark the media request object as complete. Only bound objects can
376 * be completed.
378 void media_request_object_complete(struct media_request_object *obj);
380 #else
382 static inline int __must_check
383 media_request_lock_for_access(struct media_request *req)
385 return -EINVAL;
388 static inline void media_request_unlock_for_access(struct media_request *req)
392 static inline int __must_check
393 media_request_lock_for_update(struct media_request *req)
395 return -EINVAL;
398 static inline void media_request_unlock_for_update(struct media_request *req)
402 static inline void media_request_object_get(struct media_request_object *obj)
406 static inline void media_request_object_put(struct media_request_object *obj)
410 static inline struct media_request_object *
411 media_request_object_find(struct media_request *req,
412 const struct media_request_object_ops *ops,
413 void *priv)
415 return NULL;
418 static inline void media_request_object_init(struct media_request_object *obj)
420 obj->ops = NULL;
421 obj->req = NULL;
424 static inline int media_request_object_bind(struct media_request *req,
425 const struct media_request_object_ops *ops,
426 void *priv, bool is_buffer,
427 struct media_request_object *obj)
429 return 0;
432 static inline void media_request_object_unbind(struct media_request_object *obj)
436 static inline void media_request_object_complete(struct media_request_object *obj)
440 #endif
442 #endif