Hack to workaround misbehaving CH340 drivers on Windows #200
+47
−22
Add this suggestion to a batch that can be applied as a single commit.
This suggestion is invalid because no changes were made to the code.
Suggestions cannot be applied while the pull request is closed.
Suggestions cannot be applied while viewing a subset of changes.
Only one suggestion per line can be applied in a batch.
Add this suggestion to a batch that can be applied as a single commit.
Applying suggestions on deleted lines is not supported.
You must change the existing code in this line in order to create a valid suggestion.
Outdated suggestions cannot be applied.
This suggestion has been applied or marked resolved.
Suggestions cannot be applied from pending reviews.
Suggestions cannot be applied on multi-line comments.
Suggestions cannot be applied while the pull request is queued to merge.
Suggestion cannot be applied right now. Please check back later.
It seems that the CH340 driver does not correctly support the
SetCommTimeouts
system call.In particular calling
SetCommTimeouts
with the following settingsshould set timeout to infinity and block the
Read
call until at least one byte is received. In reality, the CH340 behaves like there is no timeout, returning the available data immediately (or an empty read if there is no data). This violates theSetCommTimeout
definition, while all the other drivers correctly honor the timeout.Notably, the wrong behavior is present only with the "block until the first byte is received" timeout mode. Using the "block until the buffer is full" timeout mode
works as expected and the
Read
call lasts until the buffer given to theRead
call is filled. BTW this kind of timeout is not useful for us because it will force us to do an inefficient read of one byte at a time if the amount of data received is not known in advance.Another interesting discovery is that after setting a finite timeout (like for example 10 seconds):
the CH340 behaves correctly and waits 10 seconds for the first byte to come in.
This made me think that maybe it's the highest value
0xFFFFFFFE
that is not correctly interpreted by the driver, so I started searching the maximum-working-value, and after bisecting for a bit, I've found the "magic" number that finally makes the driver behave again:0x7FFFFFFE
. This makes me think that the bug in the driver is due to a signed-vs-unsigned comparison.In any case, reducing the "infinity" value from
0xFFFFFFFE
to0x7FFFFFFE
does solve the problem, and it keeps the library compatible with the other serial converters. Actually, it does halve the "infinity" timeout from circa 50 days (FFFFFFFE milliseconds) to 25 days (7FFFFFFE milliseconds), BTW I've added another patch that will transparently repeat theRead
if this very long timeout should ever happen.