c26cee817f
Add the ability for the USB FunctionFS (FFS) gadget driver to be able to create Device Firmware Upgrade (DFU) functional descriptors. [1] This patch allows implementation of DFU in userspace using the FFS gadget. The DFU protocol uses the control pipe (ep0) for all messaging so only the addition of the DFU functional descriptor is needed in the kernel driver. The DFU functional descriptor is written to the ep0 file along with any other descriptors during FFS setup. DFU requires an interface descriptor followed by the DFU functional descriptor. This patch includes documentation of the added descriptor for DFU and conversion of some existing documentation to kernel-doc format so that it can be included in the generated docs. An implementation of DFU 1.1 that implements just the runtime descriptor using the FunctionFS gadget (with rebooting into u-boot for DFU mode) has been tested on an i.MX8 Nano. An implementation of DFU 1.1 that implements both runtime and DFU mode using the FunctionFS gadget has been tested on Xilinx Zynq UltraScale+. Note that for the best performance of firmware update file transfers, the userspace program should respond as quick as possible to the setup packets. [1] https://www.usb.org/sites/default/files/DFU_1.1.pdf Signed-off-by: David Sands <david.sands@biamp.com> Co-developed-by: Chris Wulff <crwulff@gmail.com> Signed-off-by: Chris Wulff <crwulff@gmail.com> Link: https://lore.kernel.org/r/20240811000004.1395888-2-crwulff@gmail.com Signed-off-by: Greg Kroah-Hartman <gregkh@linuxfoundation.org>
107 lines
4.6 KiB
ReStructuredText
107 lines
4.6 KiB
ReStructuredText
====================
|
|
How FunctionFS works
|
|
====================
|
|
|
|
Overview
|
|
========
|
|
|
|
From kernel point of view it is just a composite function with some
|
|
unique behaviour. It may be added to an USB configuration only after
|
|
the user space driver has registered by writing descriptors and
|
|
strings (the user space program has to provide the same information
|
|
that kernel level composite functions provide when they are added to
|
|
the configuration).
|
|
|
|
This in particular means that the composite initialisation functions
|
|
may not be in init section (ie. may not use the __init tag).
|
|
|
|
From user space point of view it is a file system which when
|
|
mounted provides an "ep0" file. User space driver need to
|
|
write descriptors and strings to that file. It does not need
|
|
to worry about endpoints, interfaces or strings numbers but
|
|
simply provide descriptors such as if the function was the
|
|
only one (endpoints and strings numbers starting from one and
|
|
interface numbers starting from zero). The FunctionFS changes
|
|
them as needed also handling situation when numbers differ in
|
|
different configurations.
|
|
|
|
For more information about FunctionFS descriptors see :doc:`functionfs-desc`
|
|
|
|
When descriptors and strings are written "ep#" files appear
|
|
(one for each declared endpoint) which handle communication on
|
|
a single endpoint. Again, FunctionFS takes care of the real
|
|
numbers and changing of the configuration (which means that
|
|
"ep1" file may be really mapped to (say) endpoint 3 (and when
|
|
configuration changes to (say) endpoint 2)). "ep0" is used
|
|
for receiving events and handling setup requests.
|
|
|
|
When all files are closed the function disables itself.
|
|
|
|
What I also want to mention is that the FunctionFS is designed in such
|
|
a way that it is possible to mount it several times so in the end
|
|
a gadget could use several FunctionFS functions. The idea is that
|
|
each FunctionFS instance is identified by the device name used
|
|
when mounting.
|
|
|
|
One can imagine a gadget that has an Ethernet, MTP and HID interfaces
|
|
where the last two are implemented via FunctionFS. On user space
|
|
level it would look like this::
|
|
|
|
$ insmod g_ffs.ko idVendor=<ID> iSerialNumber=<string> functions=mtp,hid
|
|
$ mkdir /dev/ffs-mtp && mount -t functionfs mtp /dev/ffs-mtp
|
|
$ ( cd /dev/ffs-mtp && mtp-daemon ) &
|
|
$ mkdir /dev/ffs-hid && mount -t functionfs hid /dev/ffs-hid
|
|
$ ( cd /dev/ffs-hid && hid-daemon ) &
|
|
|
|
On kernel level the gadget checks ffs_data->dev_name to identify
|
|
whether its FunctionFS is designed for MTP ("mtp") or HID ("hid").
|
|
|
|
If no "functions" module parameters is supplied, the driver accepts
|
|
just one function with any name.
|
|
|
|
When "functions" module parameter is supplied, only functions
|
|
with listed names are accepted. In particular, if the "functions"
|
|
parameter's value is just a one-element list, then the behaviour
|
|
is similar to when there is no "functions" at all; however,
|
|
only a function with the specified name is accepted.
|
|
|
|
The gadget is registered only after all the declared function
|
|
filesystems have been mounted and USB descriptors of all functions
|
|
have been written to their ep0's.
|
|
|
|
Conversely, the gadget is unregistered after the first USB function
|
|
closes its endpoints.
|
|
|
|
DMABUF interface
|
|
================
|
|
|
|
FunctionFS additionally supports a DMABUF based interface, where the
|
|
userspace can attach DMABUF objects (externally created) to an endpoint,
|
|
and subsequently use them for data transfers.
|
|
|
|
A userspace application can then use this interface to share DMABUF
|
|
objects between several interfaces, allowing it to transfer data in a
|
|
zero-copy fashion, for instance between IIO and the USB stack.
|
|
|
|
As part of this interface, three new IOCTLs have been added. These three
|
|
IOCTLs have to be performed on a data endpoint (ie. not ep0). They are:
|
|
|
|
``FUNCTIONFS_DMABUF_ATTACH(int)``
|
|
Attach the DMABUF object, identified by its file descriptor, to the
|
|
data endpoint. Returns zero on success, and a negative errno value
|
|
on error.
|
|
|
|
``FUNCTIONFS_DMABUF_DETACH(int)``
|
|
Detach the given DMABUF object, identified by its file descriptor,
|
|
from the data endpoint. Returns zero on success, and a negative
|
|
errno value on error. Note that closing the endpoint's file
|
|
descriptor will automatically detach all attached DMABUFs.
|
|
|
|
``FUNCTIONFS_DMABUF_TRANSFER(struct usb_ffs_dmabuf_transfer_req *)``
|
|
Enqueue the previously attached DMABUF to the transfer queue.
|
|
The argument is a structure that packs the DMABUF's file descriptor,
|
|
the size in bytes to transfer (which should generally correspond to
|
|
the size of the DMABUF), and a 'flags' field which is unused
|
|
for now. Returns zero on success, and a negative errno value on
|
|
error.
|