Skip to content

cnlohr/rv003usb

Folders and files

NameName
Last commit message
Last commit date

Latest commit

Β 
Β 
Β 
Β 
Β 
Β 
Β 
Β 
Β 
Β 
Β 
Β 
Β 
Β 
Β 
Β 
Β 
Β 
Β 
Β 
Β 
Β 
Β 
Β 
Β 
Β 
Β 
Β 
Β 

Repository files navigation

Using USB on the CH32V003

Ever wanted a 10 cent USB-capable processor? Well, look no further. RV003USB allows you to, in firmware connect a 10 cent RISC-V to a computer with USB. It is fundamentally just a pin-change interrupt routine that runs in assembly, and some C code to handle higher level functionality.

It's small!

The bootloader version of the tool can create a HID device, enumerate and allow execution of code on-device in 1,920 bytes of code. Basic HID setups are approximately 2kB, like the Joystick demo.

It's simple!

The core assembly code is very basic, having both an interrupt and code to send. And only around 250 lines of C code to facilitate the rest of the USB protocol.

It's adaptable!

The core assembly code is standardized, and there is also a c file code, to handle different functionality with hid feature requests, control setup events, interrupt endpoints (send and receive) are all done in.

It shows both how to be a normal USB device, as well as how to write programs to run on your PC that can talk to your USB device.

It requires very little hardware!

Example Schematic

The reason why D+/D- is "flipped" here verses usb_config.h is because for USB low-speed, the D+/D- lines are swapped. It is frustratingly unintuitive.

  • U1 is a CH32V003
  • J1 is a USB type C connector.
  • R1 1.5k 5% is required under all configurations, though it may connect D- to DPU or VCC if the intent is for the part to always be on-bus.
  • U2, C1, C2 are used to run the CH32V003 at 3.3V. For USB it should not be run beyond 3.6V.
  • R2, R3 5.1k resistors are only needed if using a type C connector. If using a USB C host the host will not provide power without these.

Optionally 33 or 47 ohm resistors may be placed in-series with D+ and D- to the port to reduce noise and help protect the chip.

It's got demos!

Here are the following demos and their statuses...

Example Description ❖ 🐧
demo_gamepad Extremely simple example, 2-axis gamepad + 8 buttons βœ… βœ…
demo_composite_hid Mouse and Keyboard in one device βœ… βœ…
demo_hidapi Write code to directly talk to your project with fully-formed messages. (Works on Android) βœ… βœ…
bootloader CH32V003 Self-bootloader, able to flash itself with minichlink βœ… βœ…

And the following largely incomplete, but proof-of-concept projects:

Example Description ❖ 🐧
cdc_exp Enumerate as a USB Serial port and send and receive Data (incomplete, very simple) ⚠️ βœ…
demo_midi MIDI-IN and MIDI-OUT βœ… βœ…
test_ethernet RNDIS Device β›” βœ…

Note: CDC In windows likely CAN work, but I can't figure out how to do it. Linux explicitly blacklists all low-speed USB Ethernet that I could find. The MIDI example only demonstrates MIDI-OUT.

It's built on ch32v003fun

ch32v003fun is a minimal development SDK of sorts for the CH32V003, allowing for maximum flexability without needing lots of code surrounding a HAL.

General developer notes

Care surrounding interrupts and critical sections.

You are allowed to use interrupts and have critical sections, however, you should keep critical sections to approximately 40 or fewer cycles if possible. Additionally if you are going to be using interrupts that will take longer than about 40 cycles to execute, you must enable preemption on that interrup. For an example of how that works you can check the ws2812b SPI DMA driver in ch32v003fun. The external pin-chane-interrupt must be the highest priority. And it must never be preempted. While it's OK to have a short delay before it is fired, interrupting the USB reception code mid-transfer is prohibited.

It's still in beta.

This project is not ready for prime time, though it is sort of in a beta phase. Proof of concept works, lots of demos work. Maybe you can help out!

βœ… Able to go on-bus and receive USB frames
βœ… Compute CRC in-line while receiving data
βœ… Bit Stuffing (Works, tested)
βœ… Sending USB Frames
βœ… High Level USB Stack in C
βœ… Make USB timing more precise.
βœ… Use SE0 1ms ticks to tune HSItrim
βœ… Rework sending code to send tokens/data separately.
βœ… Fix CRC Code
βœ… Make minichlink able to use bootloader.
βœ… Optimize high-level stack.
βœ… Fit in bootloader (NOTE: Need to tighten more)
βœ… Do basic retiming, at least in preamble to improve timing.
βœ… Use HID custom messages.
βœ… Improve sync sled. I.e. coarse and fine sledding.
βœ… Abort on non-8-bit-aligned-frames.
πŸ”³ Make more demos
πŸ”³ API For self-flashing + printf from bootloader
πŸ”³ Improve timing on send, for CRC bits. Currently we are off by about 6 cycles total.
❕ Further optimize Send/Receive PHY code. (Please help)
⚠️ Enable improved retiming (Requires a few more cycles) (Please help!)
⚠️ Arduino support (someone else will have to take this on)

pFad - Phonifier reborn

Pfad - The Proxy pFad of © 2024 Garber Painting. All rights reserved.

Note: This service is not intended for secure transactions such as banking, social media, email, or purchasing. Use at your own risk. We assume no liability whatsoever for broken pages.


Alternative Proxies:

Alternative Proxy

pFad Proxy

pFad v3 Proxy

pFad v4 Proxy