从内核的角度来看,它只是一个具有一些独特行为的复合功能。只有在用户空间驱动程序通过写入描述符和字符串注册后,它才能被添加到 USB 配置中(用户空间程序必须提供与内核级复合功能在添加到配置中时提供的相同信息)。
1 2 3 4 5 6
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).
这特别意味着复合初始化函数可能不会在 init 部分(即可能不会使用 __init 标签)。
1 2
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.
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.
当所有文件关闭时,该功能会自动禁用。
1
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::
1 2 3 4 5
$ 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 ) &
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.
只有在所有声明的功能文件系统都已挂载并且所有功能的 USB 描述符都已写入其 ep0 时,gadget才会被注册。
1 2 3
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.
相反,第一个 USB 功能关闭其端点后,gadget会被注销。
1 2
Conversely, the gadget is unregistered after the first USB function closes its endpoints.