1 // SPDX-License-Identifier: GPL-2.0
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 <hans.verkuil@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>
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
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,
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
;
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
;
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
;
75 #ifdef CONFIG_MEDIA_CONTROLLER
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
)
92 spin_lock_irqsave(&req
->lock
, flags
);
93 if (req
->state
== MEDIA_REQUEST_STATE_COMPLETE
) {
97 spin_unlock_irqrestore(&req
->lock
, flags
);
103 * media_request_unlock_for_access - Unlock a request previously locked for
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
)
115 spin_lock_irqsave(&req
->lock
, flags
);
116 if (!WARN_ON(!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
)
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
++;
145 spin_unlock_irqrestore(&req
->lock
, flags
);
151 * media_request_unlock_for_update - Unlock a request previously locked for
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
)
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
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
,
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
);
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
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
;
272 struct media_request
*req
;
273 struct list_head list
;
278 #ifdef CONFIG_MEDIA_CONTROLLER
281 * media_request_object_get - Get a media request 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
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
,
323 * media_request_object_init - Initialise a media request 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.
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
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
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
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
375 * Mark the media request object as complete. Only bound objects can
378 void media_request_object_complete(struct media_request_object
*obj
);
382 static inline int __must_check
383 media_request_lock_for_access(struct media_request
*req
)
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
)
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
,
418 static inline void media_request_object_init(struct media_request_object
*obj
)
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
)
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
)