- 
                Notifications
    You must be signed in to change notification settings 
- Fork 147
Rework rp2xxx USB driver #658
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
base: main
Are you sure you want to change the base?
Conversation
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Great rework! The USB drive code is much cleaner now. Here are my initial comments. I'll try to more detailed review later.
        
          
                core/src/core/usb/cdc.zig
              
                Outdated
          
        
      | if (this.rx_buf) |rx| { | ||
| const len = @min(rx.len, dst.len); | ||
| // TODO: please fixme: https://github.com/ZigEmbeddedGroup/microzig/issues/452 | ||
| std.mem.copyForwards(u8, dst, rx[0..len]); | 
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Regarding those "TODO #452" memcpy should be part of DeviceInterface I guess as it is in TinyUSB and CherryUSB projects
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I disagree, the @memcpy  builtin is meant to provide a good implementation regardless of the target.
After checking on actual hardware, it seems like the issue is gone, maybe because of the recent rom changes?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Here is issue description in:
- CherryUSB: Change all memcpy to usb_memcpy cherry-embedded/CherryUSB#297 memcpy implementation in newlib (arm toolchain) is wrong.
- TinyUSB: https://github.com/liamfraser/tinyusb/blob/31a979a6cccc2d4dc0a86474643710ae3b7f34fe/src/portable/raspberrypi/rp2040/rp2040_usb.c#L57
- PicoSDK: https://github.com/raspberrypi/pico-sdk/blob/ee68c78d0afae2b69c03ae1a72bf5cc267a2d94c/src/rp2_common/pico_mem_ops/CMakeLists.txt#L15 (beware commpiler memcpy cannot be used for unaligned copies in peripheral space)
Don't know what to think about it. At least I would extract it to some inline usb_memcpy function and use std.mem.copyForwards or @memcpy inside for now.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
The usb DPRAM behaves like regular memory unlike most peripheral memory.
 
As per zig documentation on @memcpy both source and destination can have any alignment.
I can't comment on how good the memcpy implementation is, but judging from the lack of HardFault exceptions raised it must be correctly handling unaligned data. I also disassembled the .elf file of the hid example and the implementation of __aeabi_memcpy uses ldrb and strb instructions, which work on unaligned data.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Not sure how I got this mixed up, but the whole time I was thinking it was the RP2040 that had problems with memcpy. #452 of course clearly states the bug affects RP2350...
And unsurprisingly the datasheet documents this:
 
This is rather annoying. How are we supposed to guarantee that the optimizer does not use unaligned LDR/LDRH? I originally wanted to avoid needlessly copying the buffers but that seems like the simplest option now.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
So we must read/write data to some intermediate buffer (in main memory) to make sure that aligned nonoptimized memcpy operation is used right?
| @piotrfila, what toolchain version are you using, and which boards were tested? I am using Zig 0.14.1 on Windows and tried to run this code on the RP2040 and RP2350 (Arm cores). For the RP2350, I get some Windows USB errors, and the RP2040 is somewhat unstable—sometimes it works, and sometimes it doesn't. | 
| I am also using 0.14.1, but I have done all the testing under Linux, where RP2040 seems to work fine. | 
| I'm curious if you've looked at somehow making this new implementation compatible at some level with tinyUSB. I've been playing with linking the current main branch USB stuff up with a tinyUSB static library - it seems possible to me but I'm in over my head enough to know I'm likely not understanding everything right yet. In my view, one of the major benefits of Zig is that we can leverage existing c libraries and this seems like a good opportunity to not have to re-invent the wheel. | 
| @Kytezign Yes, TinyUSB and CherryUSB ports will probably be the default for the majority of users. The MicroZig custom USB stack implementation is mainly an experiment to see how far we can go with a custom implementation and whether we can design a more user-friendly API by leveraging Zig language features. | 
| 
 Makes sense.  Are TinyUSB or CherryUSB ports already being worked on or available? | 
| 
 Nothing I am aware of. Would be nice to have such port as a part of MicroZig along with other libraries like lwIP. | 
| 
 From my limited study of it, one easier route to enable this - at least for tinyUSB -  is if the standardized abstract usb interface was compatible with the tinyUSB interface.  That way all usb ports in microzig would be compatible. | 
This PR is my shot at a better USB driver implementation.
Key changes:
I'm open to suggestions for changes