Re: [PATCH/RFC 5/5] usb: Add support for streams alloc/dealloc todevio.c

From: William Gulland
Date: Thu Jun 30 2011 - 14:39:17 EST


I haven't had a chance to really look into this yet, but we use usbfs and will want to use streams from userspace (from a VM we've connected a USB3 device to).

William

----- Original Message -----
From: "Sarah Sharp" <sarah.a.sharp@xxxxxxxxxxxxxxx>
To: "Tatyana Brokhman" <tlinder@xxxxxxxxxxxxxx>
Cc: greg@xxxxxxxxx, linux-usb@xxxxxxxxxxxxxxx, linux-arm-msm@xxxxxxxxxxxxxxx, balbi@xxxxxx, ablay@xxxxxxxxxxxxxx, "Amit Blay" <ablay@xxxxxxxxxxxx>, "open list" <linux-kernel@xxxxxxxxxxxxxxx>
Sent: Thursday, June 30, 2011 10:45:38 AM
Subject: Re: [PATCH/RFC 5/5] usb: Add support for streams alloc/dealloc to devio.c

On Thu, Jun 16, 2011 at 04:31:07PM +0300, Tatyana Brokhman wrote:
> Allow user space applications such as LIBUSB, to request
> streams alloc/dealloc from HCD that implements XHCI.

Hi Tatyana,

Will this patch be used to create a userspace UAS driver, or are you
envisioning other ways userspace might use streams? I ask because I
can't think of another usage of streams in currently available devices.

This patch is good to have, but I can't imagine that you'll get good
performance if you're trying to create a userspace UAS device. After
all, usbfs does some fairly stupid things, performance wise, like using
kmalloc to allocate its URB buffers instead of usb_alloc_coherent, which
may cause the userspace buffer to be copied twice if the kmalloc memory
is not DMA'able.

usbfs probably could be changed to pin the user pages and the scatter
gather URB interface to try and avoid even one copy, but that's a
performance enhancement I haven't had time to play around with. :)

> Signed-off-by: Amit Blay <ablay@xxxxxxxxxxxx>
> Signed-off-by: Tatyana Brokhman <tlinder@xxxxxxxxxxxxxx>
>
> ---
> drivers/usb/core/devio.c | 128 +++++++++++++++++++++++++++++++++++++++++-
> include/linux/usbdevice_fs.h | 5 ++
> 2 files changed, 132 insertions(+), 1 deletions(-)
>
> diff --git a/drivers/usb/core/devio.c b/drivers/usb/core/devio.c
> index 37518df..7e73e35 100644
> --- a/drivers/usb/core/devio.c
> +++ b/drivers/usb/core/devio.c
> @@ -943,6 +943,115 @@ static int proc_clearhalt(struct dev_state *ps, void __user *arg)
> return usb_clear_halt(ps->dev, pipe);
> }
>
> +static int proc_allocstreams(struct dev_state *ps, void __user *arg)

It looks like userspace doesn't have a way to specify the number of
streams USBFS needs to allocate. Not all userspace applications are
going to want to use all available streams (especially if the device
advertises the max of 65533).

You probably also need a way to communicate back to userspace how many
streams were actually allocated. The program could choose to free
streams if it doesn't get the amount it expected. (Also, it needs to
know how many streams are available, so it can set the stream ID.)

> +{
> + unsigned int ep_map;
> + int ret;
> + int intf_num;
> + struct usb_interface *intf = NULL;
> + const int max_eps = 32;
> + int max_streams = 0;
> + struct usb_host_endpoint *eps[max_eps];
> + int num_eps = 0;
> + int i;
> + unsigned int ep;
> +
> + if (get_user(ep_map, (unsigned int __user *)arg))
> + return -EFAULT;
> +
> + for (i = 0; i < max_eps; i++) {
> + if (ep_map & (0x1 << i)) {
> + /* Convert from i to ep address */
> + if (i < 16) /* IN EP */
> + ep = i | USB_ENDPOINT_DIR_MASK;
> + else /* OUT EP */
> + ep = (i - 16);
> +
> + intf_num = findintfep(ps->dev, ep);
> + if (intf_num < 0)
> + return intf_num;
> + ret = checkintf(ps, intf_num);
> + if (ret)
> + return ret;
> + intf = usb_ifnum_to_if(ps->dev, intf_num);
> + if (!intf)
> + return -ENOENT;
> +
> + if (ep & USB_ENDPOINT_DIR_MASK)
> + eps[num_eps] = ps->dev->ep_in[ep &
> + USB_ENDPOINT_NUMBER_MASK];
> + else
> + eps[num_eps] = ps->dev->ep_out[ep &
> + USB_ENDPOINT_NUMBER_MASK];
> +
> + if (!max_streams)
> + max_streams = USB_SS_MAX_STREAMS(
> + eps[num_eps]->ss_ep_comp.bmAttributes);
> +

So you're just taking the max_streams from the first endpoint that has
streams? What if other endpoints have varying numbers of max streams?
(A well-designed device probably wouldn't, but the spec doesn't disallow
it.)

All this checking is a bit redundant anyway. The xHCI driver already
loops over the endpoints that you give it, checking for the minimum
number of reported maximum streams. It will also make sure that isn't
more than the host controller supports. So you might as well just ask
for the maximum possible number of streams (65533).

> + num_eps++;
> + }
> + }
> +
> + if (!intf || !max_streams)
> + return -ENOENT;
> +
> + ret = usb_alloc_streams(intf, eps, num_eps, max_streams, GFP_KERNEL);
> + if (ret > 0)
> + return 0;
> + return ret;
> +}
> +


What if you have endpoints on an interface that don't support streams?
It looks like you're passing the xHCI driver all endpoints on a
particular interface. The call to usb_alloc_streams() will fail if any
of those endpoints don't support streams.

So it seems like you also need a way for userspace to specify which
endpoints get streams, and which endpoints have streams freed. I will
have to check, but I think the call to usb_free_streams() will fail if
any of the endpoints don't actually have streams enabled.

Sarah Sharp
--
To unsubscribe from this list: send the line "unsubscribe linux-usb" in
the body of a message to majordomo@xxxxxxxxxxxxxxx
More majordomo info at http://vger.kernel.org/majordomo-info.html
--
To unsubscribe from this list: send the line "unsubscribe linux-kernel" in
the body of a message to majordomo@xxxxxxxxxxxxxxx
More majordomo info at http://vger.kernel.org/majordomo-info.html
Please read the FAQ at http://www.tux.org/lkml/