Andrzej Pietrasiewicz
March 27, 2019
Reading time:
In the previous post I introduced you to the subject of USB gadgets implemented as machines running Linux. We also talked about modern style of USB gadget creation and integrating that with systemd. In this post, we look at how to implement your very own USB function with FunctionFS and how to integrate that with systemd.
The idea is to delegate actual USB function implementation to userspace, using a filesystem interface (read()/write() etc). This way, USB out traffic (from host) is available at a file descriptor ready for reading, and USB in traffic (to host) is accepted at a file descriptor ready for writing.
In USB all communication is performed through so called endpoints. At the device side the endpoints are (hardware) FIFO queues associated with the UDC. There are 4 types of USB endpoints: control, bulk, iso and interrupt. Control endpoint is endpoint 0 and all USB devices shall have at least endpoint 0. This is a bi-directional communication channel between the host and the device used for enumerating and then controlling the device - remember, the USB is a host centric bus. So even if a device has some new data to be sent to the host it is the host that actually asks for data and the communication required to arrange this happens on the endpoint 0. All other types of endpoints are uni-directional. Bulk endpoints are meant for tansferring (potentially) large amounts of data on a "best effort" basis, that is, as available bandwith allows, so there are no timing guarantees for bulk data.
On the other hand bulk data is guaranteed to be transmitted error-free (perhaps using re-transmission under the hood). Iso(chronous) endpoints are meant for time-sensitive data (such as audio or video) with guaranteed bandwitdh and bounded latency. Neither data delivery nor integrity is guarenteed, though, and it is on purpose: in multimedia transmission much more harm is done by non-timely data delivery rather than by occasional non-integrity or even loss of a frame. Interrupt endpoints are for non-periodic data transfers "initiated" by the device (such as mouse movements or keystrokes) and have guaranteed latency. The "initiation" by the device is in fact queuing the data on an endpoint, and then the host polls it when it considers appropriate. In USB it's a host's world. Get used to it or quit using USB.
Why am I telling you this? You need to know that each USB function requires its own set of endpoints, which must not be used at the same time by any other function. This makes UDC's endpoints a scarce resource and imposes a limit on a number of functions which can be provided by your gadget at a time (in one configuration, but you can "overcommit" by specifying multiple configurations - you still remember, that only one of them can be active at a time?). If you want to roll your own USB function, you need to know how many and what kind of endpoints you need, and, especially if you want to compose it with other functions, whether the number of available endpoints is not exceeded. Modern UDCs usually do provide enough endpoints to compose 2-3 moderately "endpoint-hungry" functions into one configuration without any problems.
Now that you know about endpoints, you can learn more about FunctionFS. From the point of view of the gadget it is yet another USB function available for composing a gadget from. But it is special, because each instance of FunctionFS provides an instance of a 'functionfs' filesystem to be mounted. If you mount it you will notice there is only one entry: ep0. Its name suggests it is associated with endpoint 0. Indeed it is. However, the good news is that you don't need to implement all the endpoint 0 handling, because most of that is already taken care of for you by the composite layer. You only need to handle control requests directed at your particular USB function. ep0 is also (ab)used to specify the endpoints and USB strings. FunctionFS is not considered ready for binding until endpoint descriptors and strings are written to ep0. The gadget as a whole is not ready for binding if any of its functions is not ready. After the descriptors and strings are written, FunctionFS creates appropriate number of ep<number> files for you to use to implement the desired data streams between the host and the device.
An example function can be found in kernel sources: tools/usb/ffs-test.c The file also contains an example of how to specify USB descriptors and strings to be written to ep0 to make FunctionFS ready. Actually I'm using its modified and simplified version, which passes to the host whatever the host writes to it. We will get to its code later in this post.
Compared to the previous scenario there are more steps involved before the gadget becomes operational:
It turns out that systemd already has means to automate writing the descriptors and strings to FunctionFS, and starting a userspace daemon! However, the remaining steps need to be taken care of.
Let's call the scheme ffs_test.scheme. It describes a minimal gadget with FunctionFS, leaving some attributes at their default values:
attrs : { idVendor = 0xABCD; idProduct = 0x1234; }; strings = ( ); functions : { ffs_loopback : { instance = "loopback"; type = "ffs"; }; }; configs = ( { id = 1; name = "c"; functions = ( { name = "ffs.loopback"; function = "ffs_loopback"; } ); } );
One important thing to note is that the instance name (here: "loopback") becomes a "device" name to be used when mounting this FunctionFS instance.
Let's call our example service unit usb-gadget-ffs.service. This unit is almost identical to the usb-gadget.service, except the ExecStart line:
ExecStart=/bin/gt load -o ffs_test.scheme ffs_test
The difference is that we tell gt to only (-o) load gadget's composition but not bind it, which would fail anyway because the FunctionFS instance is not ready yet.
Another difference is that the Type cannot be "simple" this time, because we need systemd not to schedule loading of dependent modules until gadget scheme is fully loaded - and, consequently, FunctionFS instance registered and made available for mounting. Units of type "simple" are considered completely run immediately, so we need to change to:
Type=oneshot
This change ensures that the dependent modules will be started only after full completion of the gt command.
Complete usb-gadget-ffs.service code:
[Unit] Description=Load USB gadget scheme Requires=sys-kernel-config.mount After=sys-kernel-config.mount [Service] ExecStart=/bin/gt load -o ffs_test.scheme ffs_test RemainAfterExit=yes ExecStop=/bin/gt rm -rf ffs_test Type=oneshot [Install] WantedBy=usb-gadget.target
We can use a mount unit to automatically mount our FunctionFS instance. We choose to mount it at /run/ffs_test, so the mount unit must be named run-ffs_test.mount:
[Unit] Description=Mount FunctionFS instance Requires=usb-gadget-ffs.service After=usb-gadget-ffs.service Before=ffs.socket [Mount] # "device" name (FunctionFS instance name) What=loopback Where=/run/ffs_test Type=functionfs Options=defaults TimeoutSec=5 [Install] WantedBy=usb-gadget.target
and then use systemctl enable run-ffs_test.mount
The above two units are enough to load our gadget composition into memory at UDC's appearance and mount its accompanied FunctionFS instance.
systemd supports socket units capable of listening to usb traffic directed at FunctionFS. Such a socket unit must be pointed at an already mounted FunctionFS instance. The socket unit contains ListenUSBFunction entry in its [Socket] section to specify exactly that. It also contains a Service entry pointing to the service unit associated with this unit and such a socket unit must have its associated service unit.
When the socket unit is started, it starts its associated service unit, which contains USBFunctionDescriptors and USBFunctionStrings entries. The two specify locations of files containing binary blobs representing USB descriptors and strings, respectively. The service unit's job is then to write those to ep0 of the corresponding FunctionFS instance. And here the systemd's lazy daemon start comes into play: the descriptors and strings are already passed to FunctionFS, so it becomes ready _but_ the daemon is _not_ started until some traffic directed at this FunctionFS instance happens on USB. And it is exactly the job of the socket unit to make it work. The daemon binary is specified with the usual ExecStart entry in the service unit.
Let's call it ffs.socket:
[Unit] Description=USB function fs socket Requires=run-ffs_test.mount After=run-ffs_test.mount DefaultDependencies=no [Socket] ListenUSBFunction=/run/ffs_test Service=functionfs-daemon.service # we will get to ExecStartPost later ExecStartPost=/bin/gt enable ffs_test [Install] WantedBy=usb-gadget.target
And then systemctl enable functionfs.socket.
We call it functionfs-daemon.service as per the Service entry above.
[Service] ExecStart=/root/bin/ffs-test USBFunctionDescriptors=/root/descriptors-ffs-test.bin USBFunctionStrings=/root/strings-ffs-test.bin
A question you likely ask yourself is where to get the .bin files from? include/uapi/linux/usb/functionfs.h in the kernel sources provides the format description (line 89 in v5.0-rc6). It is up to you how you create these blobs. An example of how to create the descriptors (and strings) in a C program can be found in tools/usb/ffs-test.c. Please note that you are supposed to use usb_functionfs_descs_head_v2, because the other format is now deprecated. You can modify the program so that it doesn't do anything except writing the descriptors/strings to standard output and then capture the result in a file.
Here is a hex dump of descriptors and strings blobs for a high-speed-only example funtion:
# hd descriptors-ffs-test.bin 00000000 03 00 00 00 27 00 00 00 02 00 00 00 03 00 00 00 |....'...........| 00000010 09 04 00 00 02 ff 00 00 01 07 05 81 02 00 02 00 |................| 00000020 07 05 02 02 00 02 01 |.......| 00000027 # hd strings-ffs-test.bin 00000000 02 00 00 00 1d 00 00 00 01 00 00 00 01 00 00 00 |................| 00000010 09 04 55 53 42 20 46 69 6c 74 65 72 00 |..USB Filter.| 0000001d
Please note that the descriptors are created only in the unmodified version of ffs-test.c. As promised, we will get to the modified version later in this post.
If the functionfs.socket did not contain the ExecStartPost entry, then at this point we would have a gadget ready to be bound, but not actually bound. The ExecStartPost contains a command which executes our gadget binding _after_ the service unit is started, which is the last missing piece in this puzzle. Your gadget is now composed and activated with systemd.
No, you can't. However, usually patching userspace code to be able to be passed open file descriptors by systemd is a trivial task. And to make it even easier for you to play with FunctionFS and systemd I have modified the ffs-test.c program for using with systemd (it purposedly contains some simplifications in order for you to be able to follow the code easily, so you can't take it as a full-fledged implementation):
https://gitlab.collabora.com/andrzej.p/ffs-systemd/tree/ffs-systemd
look for tools/usb/ffs-test.c on the ffs-systemd branch.
In order to test the above mentioned modified function you need the following:
Set up your gadget and connect it to the host, it should enumerate correctly. The idea is to bind generic usb serial driver at the host to our device. This results in automatic creation of ttyUSB<X> at the host side. And then whatever you write to the ttyUSB<X> you can read back from it. That's what the modified ffs-test.c does - a loopback function.
# at the host echo 0xabcd 0x1234 > /sys/bus/usb-serial/drivers/generic/new_id dmesg <...> usbserial_generic 3-1.2.1.4.3:1.0: The "generic" usb-serial driver is only for testing and one-off prototypes. usbserial_generic 3-1.2.1.4.3:1.0: Tell linux-usb@vger.kernel.org to add your device to a proper driver. usbserial_generic 3-1.2.1.4.3:1.0: generic converter detected usb 3-1.2.1.4.3: generic converter now attached to ttyUSB1 while true; do dd if=/dev/ttyUSB1 bs=1 iflag=nocache status=none 2>/dev/null; done
And still at the host, at some other console:
man -P cat bash > /dev/ttyUSB1
The contents of bash man page should appear at the console where the "while" loop runs. You can further modify ffs-test.c to e.g. capitalize all the letters, or use some encryption algorithm to cipher your text, effectively creating a hardware crypto device on USB!
In the next installment I will write about using dummy_hcd. And in yet another one we will add some systemd templatization to make the above solutions more generic.
03/12/2024
this is a test post
08/10/2024
Having multiple developers work on pre-merge testing distributes the process and ensures that every contribution is rigorously tested before…
15/08/2024
After rigorous debugging, a new unit testing framework was added to the backend compiler for NVK. This is a walkthrough of the steps taken…
01/08/2024
We're reflecting on the steps taken as we continually seek to improve Linux kernel integration. This will include more detail about the…
27/06/2024
With each board running a mainline-first Linux software stack and tested in a CI loop with the LAVA test framework, the Farm showcased Collabora's…
26/06/2024
WirePlumber 0.5 arrived recently with many new and essential features including the Smart Filter Policy, enabling audio filters to automatically…
Comments (26)
J. Rossbach:
Aug 19, 2019 at 03:11 PM
If I have multiple interface descriptors (with different interface numbers, i.e. not alternatives to each other), can I use the same FunctionFS instance to handle them all or do I need to mount one FFS instance per interface descriptor?
Reply to this comment
Reply to this comment
Andrzej Pietrasiewicz:
Aug 26, 2019 at 09:06 PM
Yes, you can use single FunctionFS instance.
Reply to this comment
Reply to this comment
J. Rossbach:
Aug 29, 2019 at 08:44 AM
Ok, but in this case, how can I distinguish EP0 events for the different interfaces? E.g. let's assume I have two interfaces each having two alternative settings (i.e. four interfaces altogether). If the alternative setting is changed for one of the interfaces, how can I determine, for which of the two interfaces this happened and which alternative has been activated? If I would use a dedicated FFS instance for each interface on the other hand, I would have dedicated EP0s and could distinguish thereby (however I still do not know how to obtain which alternative was activated).
Reply to this comment
Reply to this comment
Sam Andre:
Jul 13, 2020 at 03:55 PM
Hello and thank you for this interesting article !
i am trying to compile ffs-test and i get the following errors:
sudo gcc ffs-test.c
/usr/bin/ld: /tmp/ccJHXqOP.o: in function `start_thread_helper':
ffs-test.c:(.text+0x89c): undefined reference to `__pthread_register_cancel'
/usr/bin/ld: ffs-test.c:(.text+0x8a0): undefined reference to `pthread_testcancel'
/usr/bin/ld: ffs-test.c:(.text+0x9b4): undefined reference to `__pthread_unregister_cancel'
/usr/bin/ld: /tmp/ccJHXqOP.o: in function `start_thread':
ffs-test.c:(.text+0xa50): undefined reference to `pthread_create'
/usr/bin/ld: /tmp/ccJHXqOP.o: in function `join_thread':
ffs-test.c:(.text+0xab8): undefined reference to `pthread_join'
collect2: error: ld returned 1 exit status
Reply to this comment
Reply to this comment
SamAndre:
Jul 14, 2020 at 11:35 AM
OK i mixed up with the deprecated file.
Now i used the new ffs-test.c file
the following steps i have done
is something wrong with it or missunderstood?
----------------------------------------------
/etc/systemd/system/
-> ffs_test.scheme
-> usb-gadget-ffs.service
-> run-ffs_test.mount ( systemctl enable run-ffs_test.mount )
-> ffs.socket (systemctl enable ffs.socket )
-> functionfs-daemon.service
-------------------------------------------
used makefile from system-branch to compile ffs-test.c
moved binary ffs-test to /bin/
-------------------------------------------
no response when the pi is connected to the host
Reply to this comment
Reply to this comment
Sam Andre:
Jul 14, 2020 at 12:47 PM
OK i mixed up with the deprecated file.
Now i used the new ffs-test.c file
the following steps i have done
is something wrong with it or missunderstood?
----------------------------------------------
/etc/systemd/system/
-> ffs_test.scheme
-> usb-gadget-ffs.service
-> run-ffs_test.mount ( systemctl enable run-ffs_test.mount )
-> ffs.socket (systemctl enable ffs.socket )
-> functionfs-daemon.service
-------------------------------------------
used makefile from system-branch to compile ffs-test.c
moved binary ffs-test to /bin/
-------------------------------------------
no response when the pi is connected to the host
Reply to this comment
Reply to this comment
Andrzej Pietrasiewicz:
Jul 14, 2020 at 05:02 PM
Did you provide descriptor and string blobs?
Reply to this comment
Reply to this comment
Sam Andre:
Jul 15, 2020 at 04:19 PM
invoking ffs-test results in :
ffs-test: Cannot inherit descriptors from systemd: (-0) Success
i hope i have not missunderstood the tutorial.
compiled ffs-test should create the descriptor and string blobs right ?
Reply to this comment
Reply to this comment
Andrzej Pietrasiewicz:
Jul 16, 2020 at 03:54 PM
"Please note that the descriptors are created only in the unmodified version of ffs-test.c." so to create the descriptors you need to build the unmodified version, too. Or use a hex editor to hand-craft them.
The "Cannot inherit descriptors from systemd" message likely means that either something is mis-configured or you use systemd version which does not support FunctionFS.
Reply to this comment
Reply to this comment
SamAndre:
Jul 26, 2020 at 09:37 PM
Thanks for your answer and help! Yes there is a deeper problem/misconfiguration.
To find the problem i maked a step back and tried the ecm gadget from page 1 and i found the following problem :
I can successfully load the gadget via console with "gt load ecm.scheme ecm" ,
but when i try the systemd solution, "usb-gadget.service" reports the following error:
"usb-gadget.service - Load USB gadget scheme for ecm
Loaded: loaded (/etc/systemd/system/usb-gadget.service; disabled; vendor preset: enabled)
Active: failed (Result: exit-code) since Sun 2020-07-26 19:35:16 BST; 1h 51min ago
Main PID: 259 (code=exited, status=255/EXCEPTION)
Jul 26 19:35:13 raspberrypi systemd[1]: Started Load USB gadget scheme for ecm.
Jul 26 19:35:13 raspberrypi gt[259]: Could not find matching gadget file.
Jul 26 19:35:16 raspberrypi systemd[1]: usb-gadget.service: Main process exited, code=exited, status=255/EXCEPTION
Jul 26 19:35:16 raspberrypi systemd[1]: usb-gadget.service: Failed with result 'exit-code'.
"
Dont know why the service cant find the scheme file , they are both in the same folder !
I did a fresh installation of Raspian Buster and got the same result again.
Reply to this comment
Reply to this comment
Andrzej Pietrasiewicz:
Aug 04, 2020 at 09:34 AM
It seems you placed your gadget scheme in a location unknown to gt. By default gt expects scheme files in /etc/gt/templates, see https://github.com/kopasiak/gt/blob/a425870aa14d35357dc3bd51e3233c5b7f51bbd1/source/gt.conf#L7 and https://github.com/kopasiak/gt/blob/a425870aa14d35357dc3bd51e3233c5b7f51bbd1/source/gt.conf#L11.
Reply to this comment
Reply to this comment
Sam Andre:
Sep 16, 2020 at 11:19 PM
Thanks for your answer !
Meanwhile i got it working partly and i detected some problems.
I created the string and descriptor blobs exactly as described but i have a problem with the second Endpoint-Descriptor [07 05 02 02 00 02 01] .
The result of "USB Descriptor Dumper" shows the second Endpoint-Descriptor is configured as "0x01 ( OUT endpoint 1 )" instead of "0x02 ( OUT endpoint 2)".
I have no Idea why its being ignored. The first Endpoint-Descriptor is configured correct.
Second Question
How can i set the Device-Descriptor with Data from the String Descriptor-Table in this case?
things like:
iManufacturer = 0x01 ;
or
manufacturer = 0x01 ;
on attrs-area in the scheme-file not working!
Reply to this comment
Reply to this comment
Andrzej Pietrasiewicz:
Sep 17, 2020 at 03:26 PM
Did you properly set bNumEndpoints? If it equals, say, "1" then chances are the other descriptor is never taken into account.
As for manufacturer string, try this in the gadget scheme:
strings = (
{
lang = 0x409;
manufacturer = "Collabora";
product = "MTP Gadget";
}
);
Reply to this comment
Reply to this comment
Sebastian:
Jul 24, 2023 at 03:06 PM
I'm struggling with the same problem - after enabling verbose debug I can see that autoconfig changes my second bEndPointAddress from 0x02 to 0x01, even though 0x81 already uses EP1:
[ 23.635312] ffs_do_descs()
[ 23.638007] ffs_do_single_desc()
[ 23.641227] interface descriptor
[ 23.644448] entity INTERFACE(00)
[ 23.647663] entity STRING(01)
[ 23.650624] : Original ep desc: 07 05 81 02 00 02 01 .......
[ 23.658796] autoconfig
[ 23.661153] : Rewritten ep desc: 07 05 81 02 00 02 01 .......
[ 23.669325] ffs_do_single_desc()
[ 23.672550] endpoint descriptor
[ 23.675674] entity ENDPOINT(81)
[ 23.678806] : Original ep desc: 07 05 02 02 00 02 01 .......
[ 23.686971] autoconfig
[ 23.689326] : Rewritten ep desc: 07 05 01 02 00 02 01 .......
[ 23.697492] ffs_do_single_desc()
[ 23.700714] endpoint descriptor
[ 23.703841] entity ENDPOINT(01)
FWIW I'm creating the gadget via configfs and my user-space function is the only function of the only config. This is for a Xilinx ZynqMP using DWC3.
Any ideas what is causing this or how to further debug this would be greatly appreciated.
Reply to this comment
Reply to this comment
sam:
Apr 07, 2023 at 07:07 AM
Hello, Could you let me know how you are able to get the systemd socket inherited and the modified fs_test.c program working ?
I am doing the same step mentioned here, but I get error , which I already posted
Reply to this comment
Reply to this comment
sam:
Nov 27, 2022 at 12:46 AM
could you please give the exact source code for building the binary blob ?
Reply to this comment
Reply to this comment
stuomas:
Mar 26, 2021 at 10:50 AM
Have you tried this with a dual role USB port? I use the same systemd files like in your Linux Piter 2019 youtube video, and I'm suffering from synchronous external aborts/kernel panics often when detaching a USB stick, and extcon driver changes role from host to device.
My gadget is RNDIS+MTP.
If I disable the gadget first (gt disable ffs_mtp) and then disconnect and role changes, everything is fine. Then I have to gt enable ffs_mtp again when I want to use it. I have automatized this with udev rules, but it feels like circumventing the actual problem.
I have no idea where the problem is, libusbgx, gt (using your fork), cmtp-responder, these systemd files, USB driver, hardware? So maybe not related to this post, just wondering if you have had any similar experiences.
Reply to this comment
Reply to this comment
Marek:
Sep 10, 2021 at 07:31 AM
Thanks a lot for valuable info. I have integrated this way my custom ffs based gadget and everything woks smoothly (host is windows). I upgraded kernel from 4.12 to 5.4 and not on windows I'm seeing that device is not recognized properly. I didn't change anything in regard of gadget setup. Any ideas what else could I check? When run script which setup gadget in 5.4 it works fine. Thanks.
Reply to this comment
Reply to this comment
Marek:
Sep 10, 2021 at 04:17 PM
Found it. Seems that missing serial number in gt scheme like: serialnumber = "85246"; was missing (maybe was pre-populated on 4.12 but not on 5.4). When added this line everything works fine. Sorry for noise but maybe it helps someone ;)
Reply to this comment
Reply to this comment
sam:
Feb 19, 2023 at 07:58 PM
I am able to load the scheme and mount the path . but I get [FAILED] Failed to listen on USB function fs socket. any idea what is failing ?
[Unit]
Description=USB function fs socket
Requires=run-usb_ffs.mount
After=run-usb_ffs.mount
DefaultDependencies=no
[Socket]
ListenUSBFunction=/run/usb_ffs/
Service=usb-gadget-ffs-daemon.service
# we will get to ExecStartPost later
ExecStartPost=/usr//bin/gt enable ffs
[Install]
#WantedBy=usb-gadget.target
WantedBy=multi-user.target
Reply to this comment
Reply to this comment
sam:
Apr 04, 2023 at 01:29 PM
Hi Andrzej Pietrasiewicz,
I am following your footstep to test this systemd socket using ffs_test, earlier I didn't compile the ff_test and i was checking whether USB socket is able to listen after mounting the functionfs filesystem. Now I have installed libsystemd-devel and linux kernel headers , I compiled your modified ffs_test.c file and when executed , it throws error
root@raspberrypi:/home/pi# ./a.out
ffs-test: Cannot inherit descriptors from systemd: (-2) No such file or directory
I have read your comments about this issue and it could be related to misconfiguration or systemd does not support functionfs , could you please help me how to make it work ?
I have searched systemd source and I did not find any functionfs related configuration in systemd. Also ffs.socket fails wile listening USB . what could be the issue , I have took all your systemd service file and binary blob
Regards
Sam
Reply to this comment
Reply to this comment
sam:
Apr 05, 2023 at 08:19 AM
Here is
root@raspberrypi:~# dmesg | grep ffs.socket
[ 29.933154] systemd[1]: ffs.socket: Installed new job ffs.socket/start as 123
[ 32.073873] systemd[1]: ffs.socket: Failed to listen on sockets: Invalid argument
[ 32.073914] systemd[1]: ffs.socket: Changed dead -> failed
[ 32.075017] systemd[1]: ffs.socket: Job ffs.socket/start finished, result=failed
[ 32.076707] systemd[1]: ffs.socket: Unit entered failed state.
[ 410.076752] systemd[1]: Preset files don't specify rule for ffs.socket. Enabling.
[ 457.632350] systemd[1]: ffs.socket: Trying to enqueue job ffs.socket/start/replace
[ 457.634429] systemd[1]: ffs.socket: Installed new job ffs.socket/start as 715
[ 457.634600] systemd[1]: ffs.socket: Enqueued job ffs.socket/start as 715
[ 457.635984] systemd[1]: ffs.socket: Failed to listen on sockets: Invalid argument
[ 457.636030] systemd[1]: ffs.socket: Job ffs.socket/start finished, result=failed
[ 566.634533] systemd[1]: ffs.socket: Trying to enqueue job ffs.socket/start/replace
[ 566.636469] systemd[1]: ffs.socket: Installed new job ffs.socket/start as 776
[ 566.636565] systemd[1]: ffs.socket: Enqueued job ffs.socket/start as 776
[ 566.639484] systemd[1]: ffs.socket: Failed to listen on sockets: Invalid argument
[ 566.639551] systemd[1]: ffs.socket: Job ffs.socket/start finished, result=failed
root@raspberrypi:~# [ 2071.839885] Under-voltage detected! (0x00050005)
Reply to this comment
Reply to this comment
sam:
Apr 12, 2023 at 12:48 AM
Hi Andrzej Pietrasiewicz
Could you please reply ? I am productizing libusbgx and stuck up with systemd listening socket and failing as mentioned in above log
Regards
sam
Reply to this comment
Reply to this comment
Andrzej Pietrasiewicz:
Apr 13, 2023 at 05:35 PM
It's difficult to say what your problem is. I'd approach it step by step if I were you. So, first make sure the usb-gadget-ffs.service runs whenever a UDC becomes available. BTW, make sure you have your usb-gadget.target and it is enabled and usb-gadget-ffs.service depends on it. Then add the mount unit and see the corresponding FunctionFS instance is properly mounted. You can then try manually passing descriptors and strings to it, and manually enabling the gadget. Only once you have that working, proceed with the socket unit, paying attention to all the dependencies and their proper spelling. And make sure your systemd is at least v227.
Reply to this comment
Reply to this comment
sam:
May 28, 2023 at 01:24 PM
I have found the issue with UDC udev event, actually there is sedit command in systemd.mk file which remove all the .target
That is why usb-gadget.target is not called . after removing it , UDC event is is calling the udev script.
Regarding on mounting /run/usb_ffs, I have found several side effects. Its not mounting every time rebooted. I see that /run is created every time rebooted and its changing the permission . So I moved the mount point to /dev/usb_ffs and its enumerating the device every time reliably
Only remaining issue is socket related descriptor initialization. still debugging it .
Reply to this comment
Reply to this comment
Quan:
Jun 21, 2024 at 09:15 AM
Have you resolved this problem yet?
Reply to this comment
Reply to this comment
Add a Comment