Bitbanging USB On Low Power ARMs

M0

With the Adafruit Trinket, the Digispark, and some very clever work with the smallest microcontroller Atmel offers, it looks like the ‘in’ thing to do for embedded software developers is to bitbang the USB protocol on hardware that shouldn’t support it. There are a lot of very small ARM chips out there without USB support, so it was only a matter of time before someone was able to bitbang USB on the ARM Cortex M0+.

The board above is based on an Energy Micro EFM32ZG, a very small 24-pin QFN device with up to 32 kB of Flash and 17 GPIOs. As with all the bitbanged USB hacks, the differential data lines are attached directly to the microcontroller. A 24 MHz crystal is needed, but the team behind the project is working on using the internal RC oscillator instead.

The code is portable with minimal changes between other manufacturer’s Cortex M0+ chips, and with a little work, this could become a very, very cheap USB-programmable ARM dev board, something the community could certainly use.

18 thoughts on “Bitbanging USB On Low Power ARMs

  1. On their write-up the pinout reference with PA0 connecting to PA0 probably was supposed to be USB 5V. The sense resistor could be 10k or more to keep from damaging a 3.3V pin.

    1. >Currently PA0 needs to connect to PC0 via a 1K5 resistor PC0 to D- PC1 to D+ of the USB port.

      They connect the 1.5K pull up to PA0 so that the device can “disconnect” under software control. USB specs specified 1.5K to form a voltage divider so that one of the data lines would be a logic high. This tells the host what connection speed the device is for. 10K is a wrong value.

      1. There are 15K pull downs for the D+/D- at the host side, so the 1.5K one one of them would form a voltage divider when sourced from a 3.3V per USB spec.

        (V-USB with 5V supply requires the Zener diodes and in that case also calls for 5V pull up.)

  2. This is really great! At 24 MHz there is also enough computational power to decode USB easily in realtime.

    Getting this to work with an internal RC oscillator is probably not that easy. I guess it is possible to build something like a DPLL into the receive loop, but then you’d also need some kind of timing correction for the sending loop? Maybe with a fractional timer?

    Things are a bit easier on AVR with RC oscillator calibration. I have not yet seen an equivalent oscillator in an ARM.

    1. The code for this project uses software delay loops. It would be fairly easy to make those fractional. For instance, instead of doing
      for(t = 0; t < 10; t++ );

      you can do:
      for( ; t < 1000; t += s );
      t -= 1000;

      With s = 100 for nominal speed.

    2. There is the PLL multiplier/prescaler, but it is not fine gain enough for the job. The alternatives are the old way of unrolling the loop and varying the the individual bit time effectively implementing a NCO or put in a crystal.

      1. The AVR RC oscillator frequency can be adjusted in steps of less than 0.3% by software. This allows calibrating of the system clock to an external timing reference.

        In the case of USB it is possible to use the USB keepalive signals to match the host clock. I have not yet seen a similar calibration register in a CM0.

Leave a Reply

Please be kind and respectful to help make the comments section excellent. (Comment Policy)

This site uses Akismet to reduce spam. Learn how your comment data is processed.