On 08/05/18 12:21, Mauro Carvalho Chehab wrote:
> Em Fri, 4 May 2018 15:27:50 +0300
> Sakari Ailus <sakari.ai...@iki.fi> escreveu:
> 
>> Hi Hans,
>>
>> I've read this patch a large number of times and I think also the details
>> begin to seem sound. A few comments below.
> 
> I'm sending this after analyzing the other patches in this series,
> as this is the core of the changes. So, although I wrote the comments
> early, I wanted to read first all other patches before sending it.
> 
>>
>> On Thu, May 03, 2018 at 04:52:53PM +0200, Hans Verkuil wrote:
>>> From: Hans Verkuil <hans.verk...@cisco.com>
>>>
>>> Add initial media request support:
>>>
>>> 1) Add MEDIA_IOC_REQUEST_ALLOC ioctl support to media-device.c
>>> 2) Add struct media_request to store request objects.
>>> 3) Add struct media_request_object to represent a request object.
>>> 4) Add MEDIA_REQUEST_IOC_QUEUE/REINIT ioctl support.
>>>
>>> Basic lifecycle: the application allocates a request, adds
>>> objects to it, queues the request, polls until it is completed
>>> and can then read the final values of the objects at the time
>>> of completion. When it closes the file descriptor the request
>>> memory will be freed (actually, when the last user of that request
>>> releases the request).
>>>
>>> Drivers will bind an object to a request (the 'adds objects to it'
>>> phase), when MEDIA_REQUEST_IOC_QUEUE is called the request is
>>> validated (req_validate op), then queued (the req_queue op).
>>>
>>> When done with an object it can either be unbound from the request
>>> (e.g. when the driver has finished with a vb2 buffer) or marked as
>>> completed (e.g. for controls associated with a buffer). When all
>>> objects in the request are completed (or unbound), then the request
>>> fd will signal an exception (poll).
>>>
>>> Signed-off-by: Hans Verkuil <hans.verk...@cisco.com>
> 
> Hmm... As you're adding Copyrights from Intel/Google in this patch, that
> indicates that part of the stuff you're adding here were authored by
> others. So, you should use Co-developed-by: tag here, and get the SOBs
> from the other developers that did part of the work[1].
> 
> [1] except if your work was sponsored by Cisco, Intel and Google, but
>     I think this is not the case.

I'll check this with Sakari and Google.

> 
>>> ---
>>>  drivers/media/Makefile        |   3 +-
>>>  drivers/media/media-device.c  |  14 ++
>>>  drivers/media/media-request.c | 407 ++++++++++++++++++++++++++++++++++
>>>  include/media/media-device.h  |  16 ++
>>>  include/media/media-request.h | 244 ++++++++++++++++++++
>>>  5 files changed, 683 insertions(+), 1 deletion(-)
>>>  create mode 100644 drivers/media/media-request.c
>>>  create mode 100644 include/media/media-request.h
>>>
>>> diff --git a/drivers/media/Makefile b/drivers/media/Makefile
>>> index 594b462ddf0e..985d35ec6b29 100644
>>> --- a/drivers/media/Makefile
>>> +++ b/drivers/media/Makefile
>>> @@ -3,7 +3,8 @@
>>>  # Makefile for the kernel multimedia device drivers.
>>>  #
>>>  
>>> -media-objs := media-device.o media-devnode.o media-entity.o
>>> +media-objs := media-device.o media-devnode.o media-entity.o \
>>> +              media-request.o
>>>  
>>>  #
>>>  # I2C drivers should come before other drivers, otherwise they'll fail
>>> diff --git a/drivers/media/media-device.c b/drivers/media/media-device.c
>>> index 35e81f7c0d2f..bb6a64acd3f0 100644
>>> --- a/drivers/media/media-device.c
>>> +++ b/drivers/media/media-device.c
>>> @@ -32,6 +32,7 @@
>>>  #include <media/media-device.h>
>>>  #include <media/media-devnode.h>
>>>  #include <media/media-entity.h>
>>> +#include <media/media-request.h>
>>>  
>>>  #ifdef CONFIG_MEDIA_CONTROLLER
>>>  
>>> @@ -366,6 +367,15 @@ static long media_device_get_topology(struct 
>>> media_device *mdev,
>>>     return ret;
>>>  }
>>>  
>>> +static long media_device_request_alloc(struct media_device *mdev,
>>> +                                  struct media_request_alloc *alloc)
>>> +{
>>> +   if (!mdev->ops || !mdev->ops->req_validate || !mdev->ops->req_queue)
>>> +           return -ENOTTY;
>>> +
>>> +   return media_request_alloc(mdev, alloc);
>>> +}
>>> +
>>>  static long copy_arg_from_user(void *karg, void __user *uarg, unsigned int 
>>> cmd)
>>>  {
>>>     /* All media IOCTLs are _IOWR() */
>>> @@ -414,6 +424,7 @@ static const struct media_ioctl_info ioctl_info[] = {
>>>     MEDIA_IOC(ENUM_LINKS, media_device_enum_links, 
>>> MEDIA_IOC_FL_GRAPH_MUTEX),
>>>     MEDIA_IOC(SETUP_LINK, media_device_setup_link, 
>>> MEDIA_IOC_FL_GRAPH_MUTEX),
>>>     MEDIA_IOC(G_TOPOLOGY, media_device_get_topology, 
>>> MEDIA_IOC_FL_GRAPH_MUTEX),
>>> +   MEDIA_IOC(REQUEST_ALLOC, media_device_request_alloc, 0),
>>>  };
>>>  
>>>  static long media_device_ioctl(struct file *filp, unsigned int cmd,
>>> @@ -686,6 +697,8 @@ void media_device_init(struct media_device *mdev)
>>>     INIT_LIST_HEAD(&mdev->pads);
>>>     INIT_LIST_HEAD(&mdev->links);
>>>     INIT_LIST_HEAD(&mdev->entity_notify);
>>> +
>>> +   mutex_init(&mdev->req_queue_mutex);
>>>     mutex_init(&mdev->graph_mutex);
>>>     ida_init(&mdev->entity_internal_idx);
>>>  
>>> @@ -699,6 +712,7 @@ void media_device_cleanup(struct media_device *mdev)
>>>     mdev->entity_internal_idx_max = 0;
>>>     media_graph_walk_cleanup(&mdev->pm_count_walk);
>>>     mutex_destroy(&mdev->graph_mutex);
>>> +   mutex_destroy(&mdev->req_queue_mutex);
>>>  }
>>>  EXPORT_SYMBOL_GPL(media_device_cleanup);
>>>  
>>> diff --git a/drivers/media/media-request.c b/drivers/media/media-request.c
>>> new file mode 100644
>>> index 000000000000..c216c4ab628b
>>> --- /dev/null
>>> +++ b/drivers/media/media-request.c
>>> @@ -0,0 +1,407 @@
>>> +// SPDX-License-Identifier: GPL-2.0-only
>>> +/*
>>> + * Media device request objects
>>> + *
>>> + * Copyright 2018 Cisco Systems, Inc. and/or its affiliates. All rights 
>>> reserved.
>>> + * Copyright (C) 2018 Intel Corporation
>>> + * Copyright (C) 2018 Google, Inc.
>>> + *
>>> + * Author: Hans Verkuil <hans.verk...@cisco.com>
>>> + * Author: Sakari Ailus <sakari.ai...@linux.intel.com>
>>> + */
>>> +
>>> +#include <linux/anon_inodes.h>
>>> +#include <linux/file.h>
>>> +
>>> +#include <media/media-device.h>
>>> +#include <media/media-request.h>
>>> +
>>> +static const char * const request_state[] = {
>>> +   [MEDIA_REQUEST_STATE_IDLE]       = "idle",
>>> +   [MEDIA_REQUEST_STATE_VALIDATING] = "validating",
>>> +   [MEDIA_REQUEST_STATE_QUEUED]     = "queued",
>>> +   [MEDIA_REQUEST_STATE_COMPLETE]   = "complete",
>>> +   [MEDIA_REQUEST_STATE_CLEANING]   = "cleaning",
>>> +};
>>> +
>>> +static const char *
>>> +media_request_state_str(enum media_request_state state)
>>> +{
>>> +   if (WARN_ON(state >= ARRAY_SIZE(request_state)))
>>> +           return "invalid";
>>> +   return request_state[state];
>>> +}
>>> +
>>> +static void media_request_clean(struct media_request *req)
>>> +{
>>> +   struct media_request_object *obj, *obj_safe;
>>> +
>>> +   WARN_ON(atomic_read(&req->state) != MEDIA_REQUEST_STATE_CLEANING);
>>> +
>>> +   list_for_each_entry_safe(obj, obj_safe, &req->objects, list) {
>>> +           media_request_object_unbind(obj);
>>> +           media_request_object_put(obj);
>>> +   }
>>> +
>>> +   req->num_incomplete_objects = 0;  
>>
>> The number of incomplete objects should be already zero here. I'd think
>> that a different number would suggest that something has gone very wrong
>> and should be complained about. How about adding
>> WARN_ON(req->num_incomplete_objects) above this line? 
>>
>>> +   wake_up_interruptible_all(&req->poll_wait);
>>> +}
>>> +
>>> +static void media_request_release(struct kref *kref)
>>> +{
>>> +   struct media_request *req =
>>> +           container_of(kref, struct media_request, kref);
>>> +   struct media_device *mdev = req->mdev;
>>> +
>>> +   dev_dbg(mdev->dev, "request: release %s\n", req->debug_str);
>>> +
>>> +   atomic_set(&req->state, MEDIA_REQUEST_STATE_CLEANING);
>>> +
>>> +   media_request_clean(req);
>>> +
>>> +   if (mdev->ops->req_free)
>>> +           mdev->ops->req_free(req);
>>> +   else
>>> +           kfree(req);
>>> +}
>>> +
>>> +void media_request_put(struct media_request *req)
>>> +{
>>> +   kref_put(&req->kref, media_request_release);
>>> +}
>>> +EXPORT_SYMBOL_GPL(media_request_put);
>>> +
>>> +static int media_request_close(struct inode *inode, struct file *filp)
>>> +{
>>> +   struct media_request *req = filp->private_data;
>>> +
>>> +   media_request_put(req);
>>> +   return 0;
>>> +}
>>> +
>>> +static unsigned int media_request_poll(struct file *filp,
>>> +                                  struct poll_table_struct *wait)
>>> +{
>>> +   struct media_request *req = filp->private_data;
>>> +   unsigned long flags;
>>> +   unsigned int ret = 0;
>>> +   enum media_request_state state;
>>> +
>>> +   if (!(poll_requested_events(wait) & POLLPRI))
>>> +           return 0;
>>> +
>>> +   spin_lock_irqsave(&req->lock, flags);
>>> +   state = atomic_read(&req->state);
>>> +
>>> +   if (state == MEDIA_REQUEST_STATE_COMPLETE) {
>>> +           ret = POLLPRI;
>>> +           goto unlock;
>>> +   }
>>> +   if (state != MEDIA_REQUEST_STATE_QUEUED) {
>>> +           ret = POLLERR;
>>> +           goto unlock;
>>> +   }
>>> +
>>> +   poll_wait(filp, &req->poll_wait, wait);
>>> +
>>> +unlock:
>>> +   spin_unlock_irqrestore(&req->lock, flags);
>>> +   return ret;
>>> +}
>>> +
>>> +static long media_request_ioctl_queue(struct media_request *req)
>>> +{
>>> +   struct media_device *mdev = req->mdev;
>>> +   enum media_request_state state;
>>> +   unsigned long flags;
>>> +   int ret = 0;  
>>
>> ret is unconditionally assigned below, no need to initialise here.
>>
>>> +
>>> +   dev_dbg(mdev->dev, "request: queue %s\n", req->debug_str);
>>> +
>>> +   /*
>>> +    * Ensure the request that is validated will be the one that gets queued
>>> +    * next by serialising the queueing process. This mutex is also used
>>> +    * to serialize with canceling a vb2 queue and with setting values such
>>> +    * as controls in a request.
>>> +    */
>>> +   mutex_lock(&mdev->req_queue_mutex);
>>> +
>>> +   spin_lock_irqsave(&req->lock, flags);
>>> +   state = atomic_cmpxchg(&req->state, MEDIA_REQUEST_STATE_IDLE,
>>> +                          MEDIA_REQUEST_STATE_VALIDATING);
>>> +   spin_unlock_irqrestore(&req->lock, flags);
> 
> It looks weird to serialize access to it with a mutex, a spin lock and 
> an atomic call.
> 
> IMHO, locking is still an issue here. I would love to test the 
> locks with some tool that would randomize syscalls, issuing close(),
> poll() and read() at wrong times and inverting the order of some calls, 
> in order to do some empiric test that all locks are at the right places.
> 
> Complex locking schemas like that usually tend to cause a lot of
> troubles.

Reqv15 reverts back to using spin locks only for this, so this is easier
to understand in v15. Mixing spin locks and atomic was not a good idea.

> 
>>> +   if (state != MEDIA_REQUEST_STATE_IDLE) {
>>> +           dev_dbg(mdev->dev,
>>> +                   "request: unable to queue %s, request in state %s\n",
>>> +                   req->debug_str, media_request_state_str(state));
>>> +           mutex_unlock(&mdev->req_queue_mutex);
>>> +           return -EBUSY;
>>> +   }
>>> +
>>> +   ret = mdev->ops->req_validate(req);
>>> +
>>> +   /*
>>> +    * If the req_validate was successful, then we mark the state as QUEUED
>>> +    * and call req_queue. The reason we set the state first is that this
>>> +    * allows req_queue to unbind or complete the queued objects in case
>>> +    * they are immediately 'consumed'. State changes from QUEUED to another
>>> +    * state can only happen if either the driver changes the state or if
>>> +    * the user cancels the vb2 queue. The driver can only change the state
>>> +    * after each object is queued through the req_queue op (and note that
>>> +    * that op cannot fail), so setting the state to QUEUED up front is
>>> +    * safe.
>>> +    *
>>> +    * The other reason for changing the state is if the vb2 queue is
>>> +    * canceled, and that uses the req_queue_mutex which is still locked
>>> +    * while req_queue is called, so that's safe as well.
>>> +    */
>>> +   atomic_set(&req->state,
>>> +              ret ? MEDIA_REQUEST_STATE_IDLE : MEDIA_REQUEST_STATE_QUEUED);
> 
> Why are you changing state also when ret fails?

Because the state during the validation was set to STATE_VALIDATING. On
failure that state has to revert back to IDLE.

> 
> Also, why you had to use a spin lock earlier in this function just 
> to change the req->state but you don't need to use it here?
> 
>>> +
>>> +   if (!ret)
>>> +           mdev->ops->req_queue(req);
>>> +
>>> +   mutex_unlock(&mdev->req_queue_mutex);
>>> +
>>> +   if (ret)
>>> +           dev_dbg(mdev->dev, "request: can't queue %s (%d)\n",
>>> +                   req->debug_str, ret);
>>> +   else
>>> +           media_request_get(req);
>>> +
>>> +   return ret;
>>> +}
>>> +
>>> +static long media_request_ioctl_reinit(struct media_request *req)
>>> +{
>>> +   struct media_device *mdev = req->mdev;
>>> +   unsigned long flags;
>>> +
>>> +   spin_lock_irqsave(&req->lock, flags);
>>> +   if (atomic_read(&req->state) != MEDIA_REQUEST_STATE_IDLE &&
>>> +       atomic_read(&req->state) != MEDIA_REQUEST_STATE_COMPLETE) {
>>> +           dev_dbg(mdev->dev,
>>> +                   "request: %s not in idle or complete state, cannot 
>>> reinit\n",
>>> +                   req->debug_str);
>>> +           spin_unlock_irqrestore(&req->lock, flags);
>>> +           return -EBUSY;
>>> +   }
>>> +   atomic_set(&req->state, MEDIA_REQUEST_STATE_CLEANING);
>>> +   spin_unlock_irqrestore(&req->lock, flags);
>>> +
>>> +   media_request_clean(req);
>>> +
>>> +   atomic_set(&req->state, MEDIA_REQUEST_STATE_IDLE);
>>> +
>>> +   return 0;
>>> +}
>>> +
>>> +static long media_request_ioctl(struct file *filp, unsigned int cmd,
>>> +                           unsigned long arg)
>>> +{
>>> +   struct media_request *req = filp->private_data;
>>> +
>>> +   switch (cmd) {
>>> +   case MEDIA_REQUEST_IOC_QUEUE:
>>> +           return media_request_ioctl_queue(req);
>>> +   case MEDIA_REQUEST_IOC_REINIT:
>>> +           return media_request_ioctl_reinit(req);
>>> +   default:
>>> +           return -ENOIOCTLCMD;
>>> +   }
>>> +}
>>> +
>>> +static const struct file_operations request_fops = {
>>> +   .owner = THIS_MODULE,
>>> +   .poll = media_request_poll,
>>> +   .unlocked_ioctl = media_request_ioctl,
>>> +   .release = media_request_close,
>>> +};
>>> +
>>> +int media_request_alloc(struct media_device *mdev,
>>> +                   struct media_request_alloc *alloc)
>>> +{
>>> +   struct media_request *req;
>>> +   struct file *filp;
>>> +   char comm[TASK_COMM_LEN];
>>> +   int fd;
>>> +   int ret;
>>> +
>>> +   /* Either both are NULL or both are non-NULL */
>>> +   if (WARN_ON(!mdev->ops->req_alloc ^ !mdev->ops->req_free))
>>> +           return -ENOMEM;
>>> +
>>> +   fd = get_unused_fd_flags(O_CLOEXEC);
>>> +   if (fd < 0)
>>> +           return fd;
>>> +
>>> +   filp = anon_inode_getfile("request", &request_fops, NULL, O_CLOEXEC);
>>> +   if (IS_ERR(filp)) {
>>> +           ret = PTR_ERR(filp);
>>> +           goto err_put_fd;
>>> +   }
>>> +
>>> +   if (mdev->ops->req_alloc)
>>> +           req = mdev->ops->req_alloc(mdev);
>>> +   else
>>> +           req = kzalloc(sizeof(*req), GFP_KERNEL);
>>> +   if (!req) {
>>> +           ret = -ENOMEM;
>>> +           goto err_fput;
>>> +   }
>>> +
>>> +   filp->private_data = req;
>>> +   req->mdev = mdev;
>>> +   atomic_set(&req->state, MEDIA_REQUEST_STATE_IDLE);
>>> +   req->num_incomplete_objects = 0;
>>> +   kref_init(&req->kref);
>>> +   INIT_LIST_HEAD(&req->objects);
>>> +   spin_lock_init(&req->lock);
>>> +   init_waitqueue_head(&req->poll_wait);
>>> +
>>> +   alloc->fd = fd;
>>> +
>>> +   get_task_comm(comm, current);
>>> +   snprintf(req->debug_str, sizeof(req->debug_str), "%s:%d",
>>> +            comm, fd);
>>> +   dev_dbg(mdev->dev, "request: allocated %s\n", req->debug_str);
>>> +
>>> +   fd_install(fd, filp);
>>> +
>>> +   return 0;
>>> +
>>> +err_fput:
>>> +   fput(filp);
>>> +
>>> +err_put_fd:
>>> +   put_unused_fd(fd);
>>> +
>>> +   return ret;
>>> +}
>>> +
>>> +static void media_request_object_release(struct kref *kref)
>>> +{
>>> +   struct media_request_object *obj =
>>> +           container_of(kref, struct media_request_object, kref);
>>> +   struct media_request *req = obj->req;
>>> +
>>> +   if (req)
>>> +           media_request_object_unbind(obj);
>>> +   obj->ops->release(obj);
>>> +}
>>> +
>>> +void media_request_object_put(struct media_request_object *obj)
>>> +{
>>> +   kref_put(&obj->kref, media_request_object_release);
>>> +}
>>> +EXPORT_SYMBOL_GPL(media_request_object_put);
>>> +
>>> +void media_request_object_init(struct media_request_object *obj)
>>> +{
>>> +   obj->ops = NULL;
>>> +   obj->req = NULL;
>>> +   obj->priv = NULL;
>>> +   obj->completed = false;
>>> +   INIT_LIST_HEAD(&obj->list);
>>> +   kref_init(&obj->kref);
>>> +}
>>> +EXPORT_SYMBOL_GPL(media_request_object_init);
>>> +
>>> +int media_request_object_bind(struct media_request *req,
>>> +                         const struct media_request_object_ops *ops,
>>> +                         void *priv,
>>> +                         struct media_request_object *obj)
>>> +{
>>> +   unsigned long flags;
>>> +   int ret = -EBUSY;
>>> +
>>> +   if (WARN_ON(!ops->release))
>>> +           return -EPERM;
>>> +
>>> +   obj->req = req;
>>> +   obj->ops = ops;
>>> +   obj->priv = priv;
>>> +
>>> +   spin_lock_irqsave(&req->lock, flags);
>>> +
>>> +   if (WARN_ON(atomic_read(&req->state) != MEDIA_REQUEST_STATE_IDLE))
>>> +           goto unlock;  
>>
>> Is this worth a kernel warning, or rather how the drivers / other framework
>> bits (e.g. VB2) prevent user from binding objects to non-idle requests?
>> Even if you added a similar check to the caller, the request state could
>> well change in the meantime.
>>
>> Perhaps add __must_check to the return value?
>>
>>> +
>>> +   list_add_tail(&obj->list, &req->objects);
>>> +   req->num_incomplete_objects++;
>>> +   ret = 0;
>>> +
>>> +unlock:
>>> +   spin_unlock_irqrestore(&req->lock, flags);
>>> +   return ret;
>>> +}
>>> +EXPORT_SYMBOL_GPL(media_request_object_bind);
>>> +
>>> +void media_request_object_unbind(struct media_request_object *obj)
>>> +{
>>> +   struct media_request *req = obj->req;
>>> +   enum media_request_state state;
>>> +   unsigned long flags;
>>> +   bool completed = false;
>>> +
>>> +   if (WARN_ON(!req))
>>> +           return;
>>> +
>>> +   spin_lock_irqsave(&req->lock, flags);
>>> +   list_del(&obj->list);
>>> +   obj->req = NULL;
>>> +
>>> +   state = atomic_read(&req->state);
>>> +
>>> +   if (state == MEDIA_REQUEST_STATE_COMPLETE ||
>>> +       state == MEDIA_REQUEST_STATE_CLEANING)
>>> +           goto unlock;
>>> +
>>> +   if (WARN_ON(state == MEDIA_REQUEST_STATE_VALIDATING))
>>> +           goto unlock;
>>> +
>>> +   if (WARN_ON(!req->num_incomplete_objects))
>>> +           goto unlock;
>>> +
>>> +   req->num_incomplete_objects--;
>>> +   if (state == MEDIA_REQUEST_STATE_QUEUED &&
>>> +       !req->num_incomplete_objects) {
>>> +           atomic_set(&req->state, MEDIA_REQUEST_STATE_COMPLETE);
>>> +           completed = true;
>>> +           wake_up_interruptible_all(&req->poll_wait);
>>> +   }
>>> +
>>> +unlock:
>>> +   spin_unlock_irqrestore(&req->lock, flags);
>>> +   if (obj->ops->unbind)
>>> +           obj->ops->unbind(obj);
>>> +   if (completed)
>>> +           media_request_put(req);
>>> +}
>>> +EXPORT_SYMBOL_GPL(media_request_object_unbind);
>>> +
>>> +void media_request_object_complete(struct media_request_object *obj)
>>> +{
>>> +   struct media_request *req = obj->req;
>>> +   unsigned long flags;
>>> +   bool completed = false;
>>> +
>>> +   spin_lock_irqsave(&req->lock, flags);
>>> +   if (obj->completed)
>>> +           goto unlock;
>>> +   obj->completed = true;
>>> +   if (WARN_ON(!req->num_incomplete_objects) ||
>>> +       WARN_ON(atomic_read(&req->state) != MEDIA_REQUEST_STATE_QUEUED))
>>> +           goto unlock;
>>> +
>>> +   if (!--req->num_incomplete_objects) {
>>> +           atomic_set(&req->state, MEDIA_REQUEST_STATE_COMPLETE);
>>> +           wake_up_interruptible_all(&req->poll_wait);
>>> +           completed = true;
>>> +   }
>>> +unlock:
>>> +   spin_unlock_irqrestore(&req->lock, flags);
>>> +   if (completed)
>>> +           media_request_put(req);
>>> +}
>>> +EXPORT_SYMBOL_GPL(media_request_object_complete);
>>> diff --git a/include/media/media-device.h b/include/media/media-device.h
>>> index bcc6ec434f1f..7d855823341c 100644
>>> --- a/include/media/media-device.h
>>> +++ b/include/media/media-device.h
>>> @@ -27,6 +27,7 @@
>>>  
>>>  struct ida;
>>>  struct device;
>>> +struct media_device;
>>>  
>>>  /**
>>>   * struct media_entity_notify - Media Entity Notify
>>> @@ -50,10 +51,21 @@ struct media_entity_notify {
>>>   * struct media_device_ops - Media device operations
>>>   * @link_notify: Link state change notification callback. This callback is
>>>   *          called with the graph_mutex held.
>>> + * @req_alloc: Allocate a request
>>> + * @req_free: Free a request
> 
> Please do place better descriptions there. First of all, either both
> should be used or both should be NULL - as validated by
> media_request_alloc() logic.
> 
> Second, it should be clearer that those are meant to be used when
> the driver needs to allocate a bigger struct that embeds a 
> struct media_request object on it, with should be destroyed at 
> ops->req_free() call.
> 
> Also, such embed struct should not contain a kref (multiple krefs
> at the same struct doesn't work).

Improved the comments.

> 
>>> + * @req_validate: Validate a request, but do not queue yet
>>> + * @req_queue: Queue a validated request, cannot fail. If something goes
>>> + *        wrong when queueing this request then it should be marked
>>> + *        as such internally in the driver and any related buffers
>>> + *        must eventually return to vb2 with state VB2_BUF_STATE_ERROR.
> 
> Please describe what kind of req locks (if any) can/should
> be used inside each callback.

I've mentioned that req_queue_mutex is held when calling these two ops.

> 
> 
>>>   */
>>>  struct media_device_ops {
>>>     int (*link_notify)(struct media_link *link, u32 flags,
>>>                        unsigned int notification);
>>> +   struct media_request *(*req_alloc)(struct media_device *mdev);
>>> +   void (*req_free)(struct media_request *req);
>>> +   int (*req_validate)(struct media_request *req);
>>> +   void (*req_queue)(struct media_request *req);
>>>  };
>>>  
>>>  /**
>>> @@ -88,6 +100,8 @@ struct media_device_ops {
>>>   * @disable_source: Disable Source Handler function pointer
>>>   *
>>>   * @ops:   Operation handler callbacks
>>> + * @req_queue_mutex: Serialise the MEDIA_REQUEST_IOC_QUEUE ioctl w.r.t. 
>>> this
>>> + *              media device.
> 
> This description is incomplete as it doesn't match the explanation you
> introduced at patch 00/18. Please add a complete locking description,
> as patches 00 aren't stored anywhere at the git tree, and there are lots
> of non-trivial assumptions on your locking schema.

True, updated the description. It was horrible outdated :-)

> 
>>>   *
>>>   * This structure represents an abstract high-level media device. It 
>>> allows easy
>>>   * access to entities and provides basic media device-level support. The
>>> @@ -158,6 +172,8 @@ struct media_device {
>>>     void (*disable_source)(struct media_entity *entity);
>>>  
>>>     const struct media_device_ops *ops;
>>> +
>>> +   struct mutex req_queue_mutex;
>>>  };
>>>  
>>>  /* We don't need to include pci.h or usb.h here */
>>> diff --git a/include/media/media-request.h b/include/media/media-request.h
>>> new file mode 100644
>>> index 000000000000..e39122dfd717
>>> --- /dev/null
>>> +++ b/include/media/media-request.h
>>> @@ -0,0 +1,244 @@
>>> +// SPDX-License-Identifier: GPL-2.0-only
>>> +/*
>>> + * Media device request objects
>>> + *
>>> + * Copyright 2018 Cisco Systems, Inc. and/or its affiliates. All rights 
>>> reserved.
>>> + * Copyright (C) 2018 Intel Corporation
>>> + *
>>> + * Author: Hans Verkuil <hans.verk...@cisco.com>
>>> + * Author: Sakari Ailus <sakari.ai...@linux.intel.com>
>>> + */
>>> +
>>> +#ifndef MEDIA_REQUEST_H
>>> +#define MEDIA_REQUEST_H
>>> +
>>> +#include <linux/list.h>
>>> +#include <linux/slab.h>
>>> +#include <linux/spinlock.h>
>>> +#include <linux/atomic.h>
>>> +
>>> +#include <media/media-device.h>
>>> +
>>> +/**
>>> + * enum media_request_state - media request state
>>> + *
>>> + * @MEDIA_REQUEST_STATE_IDLE:              Idle
>>> + * @MEDIA_REQUEST_STATE_VALIDATING:        Validating the request, no 
>>> state changes
>>> + *                                 allowed
>>> + * @MEDIA_REQUEST_STATE_QUEUED:            Queued
>>> + * @MEDIA_REQUEST_STATE_COMPLETE:  Completed, the request is done
>>> + * @MEDIA_REQUEST_STATE_CLEANING:  Cleaning, the request is being re-inited
>>> + */
>>> +enum media_request_state {
>>> +   MEDIA_REQUEST_STATE_IDLE,
>>> +   MEDIA_REQUEST_STATE_VALIDATING,
>>> +   MEDIA_REQUEST_STATE_QUEUED,
>>> +   MEDIA_REQUEST_STATE_COMPLETE,
>>> +   MEDIA_REQUEST_STATE_CLEANING,
>>> +};
>>> +
>>> +struct media_request_object;
>>> +
>>> +/**
>>> + * struct media_request - Media device request
>>> + * @mdev: Media device this request belongs to
>>> + * @kref: Reference count
>>> + * @debug_str: Prefix for debug messages (process name:fd)
>>> + * @state: The state of the request
>>> + * @objects: List of @struct media_request_object request objects
>>> + * @num_objects: The number of objects in the request
>>> + * @num_incompleted_objects: The number of incomplete objects in the 
>>> request
>>> + * @poll_wait: Wait queue for poll
>>> + * @lock: Serializes access to this struct
>>> + */
>>> +struct media_request {
>>> +   struct media_device *mdev;
>>> +   struct kref kref;
>>> +   char debug_str[TASK_COMM_LEN + 11];
>>> +   atomic_t state;
>>> +   struct list_head objects;
>>> +   unsigned int num_incomplete_objects;
>>> +   struct wait_queue_head poll_wait;
>>> +   spinlock_t lock;
>>> +};
>>> +
>>> +#ifdef CONFIG_MEDIA_CONTROLLER
>>> +
>>> +/**
>>> + * media_request_get - Get the media request
>>> + *
>>> + * @req: The request
>>> + *
>>> + * Get the media request.
>>> + */
>>> +static inline void media_request_get(struct media_request *req)
>>> +{
>>> +   kref_get(&req->kref);
>>> +}
>>> +
>>> +/**
>>> + * media_request_put - Put the media request
>>> + *
>>> + * @req: The request
>>> + *
>>> + * Put the media request. The media request will be released
>>> + * when the refcount reaches 0.
>>> + */
>>> +void media_request_put(struct media_request *req);
>>> +
>>> +/**
>>> + * media_request_alloc - Allocate the media request
>>> + *
>>> + * @mdev: Media device this request belongs to
>>> + * @alloc: Store the request's file descriptor in this struct
>>> + *
>>> + * Allocated the media request and put the fd in @alloc->fd.
>>> + */
>>> +int media_request_alloc(struct media_device *mdev,
>>> +                   struct media_request_alloc *alloc);
>>> +
>>> +#else
>>> +
>>> +static inline void media_request_get(struct media_request *req)
>>> +{
>>> +}
>>> +
>>> +static inline void media_request_put(struct media_request *req)
>>> +{
>>> +}
>>> +
>>> +#endif
>>> +
>>> +/**
>>> + * struct media_request_object_ops - Media request object operations
>>> + * @prepare: Validate and prepare the request object, optional.
>>> + * @unprepare: Unprepare the request object, optional.
>>> + * @queue: Queue the request object, optional.
>>> + * @unbind: Unbind the request object, optional.
>>> + * @release: Release the request object, required.
>>> + */
>>> +struct media_request_object_ops {
>>> +   int (*prepare)(struct media_request_object *object);
>>> +   void (*unprepare)(struct media_request_object *object);
>>> +   void (*queue)(struct media_request_object *object);
>>> +   void (*unbind)(struct media_request_object *object);
>>> +   void (*release)(struct media_request_object *object);
>>> +};
>>> +
>>> +/**
>>> + * struct media_request_object - An opaque object that belongs to a media
>>> + *                          request
>>> + *
>>> + * @ops: object's operations
>>> + * @priv: object's priv pointer
>>> + * @req: the request this object belongs to (can be NULL)
>>> + * @list: List entry of the object for @struct media_request
>>> + * @kref: Reference count of the object, acquire before releasing req->lock
>>> + * @completed: If true, then this object was completed.
>>> + *
>>> + * An object related to the request. This struct is embedded in the
>>> + * larger object data.
> 
> what do you mean by "the larger object data"? What struct is "the" struct?

Improved the text to clarify this.

> 
>>> + */
>>> +struct media_request_object {
>>> +   const struct media_request_object_ops *ops;
>>> +   void *priv;
>>> +   struct media_request *req;
>>> +   struct list_head list;
>>> +   struct kref kref;
>>> +   bool completed;
>>> +};
>>> +
>>> +#ifdef CONFIG_MEDIA_CONTROLLER
>>> +
>>> +/**
>>> + * media_request_object_get - Get a media request object
>>> + *
>>> + * @obj: The object
>>> + *
>>> + * Get a media request object.
>>> + */
>>> +static inline void media_request_object_get(struct media_request_object 
>>> *obj)
>>> +{
>>> +   kref_get(&obj->kref);
>>> +}
>>> +
>>> +/**
>>> + * media_request_object_put - Put a media request object
>>> + *
>>> + * @obj: The object
>>> + *
>>> + * Put a media request object. Once all references are gone, the
>>> + * object's memory is released.
>>> + */
>>> +void media_request_object_put(struct media_request_object *obj);
>>> +
>>> +/**
>>> + * media_request_object_init - Initialise a media request object
>>> + *
>>> + * Initialise a media request object. The object will be released using the
>>> + * release callback of the ops once it has no references (this function
>>> + * initialises references to one).
>>> + */
>>> +void media_request_object_init(struct media_request_object *obj);
>>> +
>>> +/**
>>> + * media_request_object_bind - Bind a media request object to a request  
>>
>> Argument documentation is missing.
>>
>> I think you should also say that "every bound object must be unbound later
>> on".
>>
>>> + */
>>> +int media_request_object_bind(struct media_request *req,
>>> +                         const struct media_request_object_ops *ops,
>>> +                         void *priv,
>>> +                         struct media_request_object *obj);
>>> +
>>> +/**
>>> + * media_request_object_unbind - Unbind a media request object
>>> + *
>>> + * @obj: The object
>>> + *
>>> + * Unbind the media request object from the request.
>>> + */
>>> +void media_request_object_unbind(struct media_request_object *obj);
>>> +
>>> +/**
>>> + * media_request_object_complete - Mark the media request object as 
>>> complete
>>> + *
>>> + * @obj: The object
>>> + *
>>> + * Mark the media request object as complete.  
>>
>> Add:
>>
>> Only bound request objects may be completed.
>>
>>> + */
>>> +void media_request_object_complete(struct media_request_object *obj);
>>> +
>>> +#else
>>> +
>>> +static inline void media_request_object_get(struct media_request_object 
>>> *obj)
>>> +{
>>> +}
>>> +
>>> +static inline void media_request_object_put(struct media_request_object 
>>> *obj)
>>> +{
>>> +}
>>> +
>>> +static inline void media_request_object_init(struct media_request_object 
>>> *obj)
>>> +{
>>> +   obj->ops = NULL;
>>> +   obj->req = NULL;
>>> +}
>>> +
>>> +static inline int media_request_object_bind(struct media_request *req,
>>> +                          const struct media_request_object_ops *ops,
>>> +                          void *priv,
>>> +                          struct media_request_object *obj)
>>> +{
>>> +   return 0;
>>> +}
>>> +
>>> +static inline void media_request_object_unbind(struct media_request_object 
>>> *obj)
>>> +{
>>> +}
>>> +
>>> +static inline void media_request_object_complete(struct 
>>> media_request_object *obj)
>>> +{
>>> +}
>>> +
>>> +#endif
>>> +
>>> +#endif  
>>
> 
> 
> 
> Thanks,
> Mauro
> 

Thank you for all the feedback!

Regards,

        Hans

Reply via email to