[<prev] [next>] [<thread-prev] [thread-next>] [day] [month] [year] [list]
Message-ID: <20250207125915.000079e4@huawei.com>
Date: Fri, 7 Feb 2025 12:59:15 +0000
From: Jonathan Cameron <Jonathan.Cameron@...wei.com>
To: Jason Gunthorpe <jgg@...dia.com>
CC: Andy Gospodarek <andrew.gospodarek@...adcom.com>, Aron Silverton
<aron.silverton@...cle.com>, Dan Williams <dan.j.williams@...el.com>, Daniel
Vetter <daniel.vetter@...ll.ch>, Dave Jiang <dave.jiang@...el.com>, David
Ahern <dsahern@...nel.org>, Andy Gospodarek <gospo@...adcom.com>, Christoph
Hellwig <hch@...radead.org>, Itay Avraham <itayavr@...dia.com>, Jiri Pirko
<jiri@...dia.com>, Jakub Kicinski <kuba@...nel.org>, Leonid Bloch
<lbloch@...dia.com>, Leon Romanovsky <leonro@...dia.com>,
<linux-cxl@...r.kernel.org>, <linux-rdma@...r.kernel.org>,
<netdev@...r.kernel.org>, Saeed Mahameed <saeedm@...dia.com>, "Nelson,
Shannon" <shannon.nelson@....com>
Subject: Re: [PATCH v4 02/10] fwctl: Basic ioctl dispatch for the character
device
On Thu, 6 Feb 2025 20:13:24 -0400
Jason Gunthorpe <jgg@...dia.com> wrote:
> Each file descriptor gets a chunk of per-FD driver specific context that
> allows the driver to attach a device specific struct to. The core code
> takes care of the memory lifetime for this structure.
>
> The ioctl dispatch and design is based on what was built for iommufd. The
> ioctls have a struct which has a combined in/out behavior with a typical
> 'zero pad' scheme for future extension and backwards compatibility.
>
> Like iommufd some shared logic does most of the ioctl marshalling and
> compatibility work and tables diatches to some function pointers for
> each unique iotcl.
>
> This approach has proven to work quite well in the iommufd and rdma
> subsystems.
>
> Allocate an ioctl number space for the subsystem.
>
> Reviewed-by: Jonathan Cameron <Jonathan.Cameron@...wei.com>
> Signed-off-by: Jason Gunthorpe <jgg@...dia.com>
Hi Jason,
Fresh read through given it's been a while.
A few really trivial things inline + one passing comment on a future
entertaining corner.
Jonathan
> M: Sebastian Reichel <sre@...nel.org>
> diff --git a/drivers/fwctl/main.c b/drivers/fwctl/main.c
> index 34946bdc3bf3d7..d561deaf2b86d8 100644
> --- a/drivers/fwctl/main.c
> +++ b/drivers/fwctl/main.c
> static int fwctl_fops_release(struct inode *inode, struct file *filp)
> {
> - struct fwctl_device *fwctl = filp->private_data;
> + struct fwctl_uctx *uctx = filp->private_data;
> + struct fwctl_device *fwctl = uctx->fwctl;
>
> + scoped_guard(rwsem_read, &fwctl->registration_lock) {
> + /*
> + * fwctl_unregister() has already removed the driver and
> + * destroyed the uctx.
Comment is a little odd given it is I think referring to why
the code that follows wouldn't run. Perhaps just add a 'may'
fwctl_unregister() may have already removed the driver and destroyed
the uctx.
> + */
> + if (fwctl->ops) {
> + guard(mutex)(&fwctl->uctx_list_lock);
> + fwctl_destroy_uctx(uctx);
> + }
> + }
> +
> + kfree(uctx);
> fwctl_put(fwctl);
> return 0;
> }
>
> @@ -71,14 +183,17 @@ _alloc_device(struct device *parent, const struct fwctl_ops *ops, size_t size)
> if (!fwctl)
> return NULL;
>
> - fwctl->dev.class = &fwctl_class;
> - fwctl->dev.parent = parent;
> -
> devnum = ida_alloc_max(&fwctl_ida, FWCTL_MAX_DEVICES - 1, GFP_KERNEL);
> if (devnum < 0)
> return NULL;
> fwctl->dev.devt = fwctl_dev + devnum;
>
> + fwctl->dev.class = &fwctl_class;
> + fwctl->dev.parent = parent;
Shunt this move back to previous patch?
> + init_rwsem(&fwctl->registration_lock);
> + mutex_init(&fwctl->uctx_list_lock);
> + INIT_LIST_HEAD(&fwctl->uctx_list);
> +
> device_initialize(&fwctl->dev);
> return_ptr(fwctl);
> }
> diff --git a/include/linux/fwctl.h b/include/linux/fwctl.h
> index 68ac2d5ab87481..93b470efb9dbc3 100644
> --- a/include/linux/fwctl.h
> +++ b/include/linux/fwctl.h
> @@ -11,7 +11,30 @@
> struct fwctl_device;
> struct fwctl_uctx;
>
> +/**
> + * struct fwctl_ops - Driver provided operations
> + *
> + * fwctl_unregister() will wait until all excuting ops are completed before it
> + * returns. Drivers should be mindful to not let their ops run for too long as
> + * it will block device hot unplug and module unloading.
A passing comment on this. Seems likely that at somepoint we'll want an
abort op to enable cancelling if the particular driver supports it
(abort background command in CXL). Anyhow, problem for another day.
> + */
> struct fwctl_ops {
> + /**
> + * @uctx_size: The size of the fwctl_uctx struct to allocate. The first
> + * bytes of this memory will be a fwctl_uctx. The driver can use the
> + * remaining bytes as its private memory.
> + */
> + size_t uctx_size;
> + /**
> + * @open_uctx: Called when a file descriptor is opened before the uctx
> + * is ever used.
> + */
> + int (*open_uctx)(struct fwctl_uctx *uctx);
> + /**
> + * @close_uctx: Called when the uctx is destroyed, usually when the FD
> + * is closed.
> + */
> + void (*close_uctx)(struct fwctl_uctx *uctx);
> };
Powered by blists - more mailing lists