Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Documentation: add multi-threading notes #45

Closed
Youw opened this issue Jul 13, 2019 · 24 comments
Closed

Documentation: add multi-threading notes #45

Youw opened this issue Jul 13, 2019 · 24 comments
Assignees
Labels
documentation Improvements or additions to documentation enhancement New feature or request

Comments

@Youw
Copy link
Member

Youw commented Jul 13, 2019

need add notes to README regarding multithreading, e.g. regarding Windows issues/non-full support
maybe some more

@Youw
Copy link
Member Author

Youw commented Jan 26, 2021

NOTE for everyone: hidapi not thread-safe in general, with a few platform-specific/usage-spacific exceptions.

Damn, I need to get my hands to describe everything in details.

@mcuee
Copy link
Member

mcuee commented Jun 15, 2021

Reference: Windows: unable to interrupt blocking hid_read call #133

@Youw Youw unpinned this issue Mar 16, 2022
@Youw Youw pinned this issue Mar 16, 2022
@Youw Youw unpinned this issue May 16, 2022
@Youw Youw pinned this issue May 16, 2022
@Qbicz Qbicz unpinned this issue Oct 13, 2022
@Qbicz Qbicz pinned this issue Oct 13, 2022
@mcuee
Copy link
Member

mcuee commented Oct 30, 2022

I see this and #133 are tied to 0.13.0 milestone. Supposedly this one will "help" #133, right?

Any activities on this one?

#146 is supposed to fix #133, but I see no activities there.

@mcuee
Copy link
Member

mcuee commented Oct 30, 2022

@Youw
Just wondering what is the plan for 0.13.0 release, say as a Christmas gift. :-)

@Youw
Copy link
Member Author

Youw commented Oct 30, 2022

0.13 already includes a lot of good improvements, so I guess making a release (before Christmas) would be a good thing regardless if it is going to include the multithreaded documentation/fixes or not

@mcuee
Copy link
Member

mcuee commented Oct 30, 2022

0.13 already includes a lot of good improvements, so I guess making a release (before Christmas) would be a good thing regardless if it is going to include the multithreaded documentation/fixes or not

I agree. Thanks.

@mcuee
Copy link
Member

mcuee commented Jan 6, 2023

@Youw

It is great that you have released hidapi 0.13. Thanks.

@mcuee
Copy link
Member

mcuee commented May 10, 2023

Reference by Youw.

As for multithreaded usage of hidapi - many projects successfully using a separate thread for hid_read/_timeout, and another thread for read/everything else.

@mcuee
Copy link
Member

mcuee commented May 10, 2023

Another reference by Youw.

client code should first stop any read/write attempts before trying to close the device as hid_close cannot not be thread-safe by design, even it it is sometime, for some implementations.

@mcuee
Copy link
Member

mcuee commented May 10, 2023

Add a stub FAQ entry here.
https://github.com/libusb/hidapi/wiki#faq

@mcuee
Copy link
Member

mcuee commented May 11, 2023

Comments by Youw here.

Oh yes, HIDAPI on macOS has an additional issue regarding thread-safety. I did encountered it in my project(s) but didn't have enough time to gather info/find a root cause. Looks like on macOS hid_init/hid_exit needs to be called in the same thread, which is an additional restriction compared to other platforms.

@mcuee
Copy link
Member

mcuee commented May 14, 2023

Reference from node-hid.
https://github.com/node-hid/node-hid#thread-safety-worker-threads-context-aware-modules

Thread safety, Worker threads, Context-aware modules
In general node-hid is not thread-safe because the underlying C-library it wraps (hidapi) is not thread-safe. However, node-hid is now reporting as minimally Context Aware to allow use in Electron v9+. Until node-hid (or hidapi) is rewritten to be thread-safe, please constrain all accesses to it via a single thread.

@k1-801
Copy link

k1-801 commented Mar 11, 2024

Oh yes, HIDAPI on macOS has an additional issue regarding thread-safety. I did encountered it in my project(s) but didn't have enough time to gather info/find a root cause. Looks like on macOS hid_init/hid_exit needs to be called in the same thread, which is an additional restriction compared to other platforms.

I have a potential solution, described fully in #666

@Masterxilo
Copy link

I found this old discussion on multithreading in hidapi maybe it helps in writing the documentation signal11/hidapi#72

@Masterxilo
Copy link

Masterxilo commented Nov 16, 2024

In my usecase, I will interact with one device on one thread always, but I want to interact with different devices on different threads simultaneously.

Are all hidapi implementations at least thread-safe enough for this?

I don't need to be able to talk to one device from multiple threads simultaneously and I would think that makes sense only very rarely anyways. I observed that for instance on Windows, multiple threads and programs talking to the same device at the same time are allowed, but the order in which the reports are received in each direction gets garbled up. Even on a conceptual level talking to one device from multiple threads doesn't make much sense.

But talking to different devices on different threads/programs seems natural and should be possible for performance reasons. It should make no difference whether I use program x to talk to device a and program y to talk to device b, or simply two threads 1&2 in a single program to talk to devices a&b.

Do the hidapi implementations guarantee that this works?

@JoergAtGithub
Copy link
Contributor

You can enumerate and open the devices in a main thread, and once opened successful, you can move the device handle into a communication thread per device, where you do all read/write ops and finally close the device.

@Masterxilo
Copy link

Thanks for the feedback @JoergAtGithub. I don't even need or want to open the devices on that main thread, but yes i will need to enumerate them once in a main thread to get the vid/pid/usb serials I am interested in.

So do the hidapi implementations guarantee that then opening the devices in the different communication threads is safe, even if, say, I were to do another enumeration again on the main thread simultaneously?

I have the feeling I saw some caching mechanisms in some of the hidapi implementations (having to do with USB characteristics?) that make me doubt this is thread-safe...

So yeah my requirement would be that there can be one main thread doing enumeration (potentially in a loop, forever), while other threads communicate with the existing devices undisturbed, one thread per device. Well maybe some threads will handle more than one device, but no device will be opened by more than one thread.

I currently make sure to do absolutely all hidapi work in a single thread and that obviously works fine, but I am afraid to break things in moving past that for performance reasons...

I would need pretty strong guarantees to be confident this works in all implementations and will continue to do so, i.e. hidapi would have to commit to and document some level of some notion of this kind of "per device" & single-enumeration thread-safety...

Note I don't need that enumeration itself is thread safe - there's no point in enumerating in more than one thread simultaneously.

@Masterxilo
Copy link

Short of that guarantee, i would imagine the only other safe option I would have is to enumerate in a main program, then spawn a separate process per device. I would imagine the operating systems guarantee this works fine when each process talks to another device and I don't imagine any hidapi implementation uses some system wide shared memory in some misguided attempt to share some cached data/state...

@fricpa
Copy link

fricpa commented Dec 5, 2024

I dug out signal11/hidapi#72 maybe it helps.

@Youw
Copy link
Member Author

Youw commented Jan 23, 2025

Sorry for ignoring this for so long.
@Masterxilo, yes it is safe to use different hidapi devices in different threads with one note: in general (croll-platform case) all hid_open/hid_close operations should be performed in the same thread (or at least serialized with) as hid_enumerate. This limitation comes from the least "safe" platform for this case - macOS. HidManager is not thread-safe to be used and it is used by all of hid_init/hid_exit/hid_enumerate/hid_open/hid_close.

So you can:

  • enumerate / open all you devices in one thread
  • use each device in its separate thread
  • close all devices in the same thread you've opened them

@Youw
Copy link
Member Author

Youw commented Jan 23, 2025

I think I'm finally getting my hands to https://github.com/libusb/hidapi/wiki/Multi%E2%80%90threading-Notes/

@Youw
Copy link
Member Author

Youw commented Jan 23, 2025

Closing in favor of https://github.com/libusb/hidapi/wiki/Multi%E2%80%90threading-Notes
If I've missed something obvious - will update later.

@Youw Youw closed this as completed Jan 23, 2025
@JoergAtGithub
Copy link
Contributor

Thank you very much for documenting this!

  • close all devices in the same thread you've opened them

Could you please explain why this is the case? This is not obvious to me.

@Youw
Copy link
Member Author

Youw commented Jan 23, 2025

Could you please explain why this is the case? This is not obvious to me.

See the NOTE about hid_close here.

close all devices in the same thread you've opened them

That's one way to enforce it.

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
documentation Improvements or additions to documentation enhancement New feature or request
Projects
None yet
Development

No branches or pull requests

6 participants