MCP2221/MCP2221A USB Converter Open Source HID Library

This is an open-source library that aims to provide a multi-platform interface solution for interacting with Microchip’s MCP2221/MCP2221A USB-to-UART and I2C/SMBus serial converters. The library supports access to various functions of the converter, including 4 GPIO pins, 3 10-bit ADCs, and 1 5-bit DAC. Unlike Microchip’s proprietary DLL library, this project aims to provide an open-source, cross-platform alternative.

This library also makes use of HIDAPI.

Supported features:

FeatureStatus
ADCSupported
DACSupported
GPIOSupported
Interrupt inputSupported
Clock reference outputSupported
USB Descriptors
(Manufacturer, product, serial, VID, PID)
Supported
I2C/SMBLimited support, WIP
Flash password protectionNot yet implemented
C++ and C# wrappersNot yet implemented

Download from GitHub
Documentation

Bits of info about the MCP2221

  • Doesn’t use a crystal, only requires 1 small capacitor when powered with 3.3V or 2 capacitors when powered with 5V.
  • Available in a hacker friendly DIP package.
  • Has a remote wake function which when used in conjunction with the interrupt input can be used to wakeup the USB host (usually a PC), just like waking up from a keyboard or mouse press.
  • The raw value of the I2C pins can also be read, allowing them to be used as an additional 2 general purpose input pins.
  • Unfortunately there are no options for enabling any internal pull-up/down resistors.
  • The MCP2221 seems to be a PIC16F1455. [Source]
  • Current consumption @ 3.3V with USB disconnected can be anywhere between 20uA and 70uA, not sure what causes such a difference. Remote wakeup needs to be disabled (default) otherwise current consumption will be about 5mA.
  • Kinda slow compared to other USB/UART converters. Even with the baud rate set to 1,000,000 its overall throughput is only about 250,000 bps. More about this below.

Unconnected pins
The usual way to deal with unconnected pins are to enable their internal pull-up/down resistors, but this chip doesn’t support them. Instead it’s probably best to set unconnected pins as output high or low.
The I2C pins are a little different since they can’t be set to outputs, though there’s still few options:

I2C PinDescription
UnconnectedBad, floating inputs will cause excessive power consumption
Connect to VDDBad, if the I2C bus is accidentally used then a short circuit will occur
Connect to groundOK, but may cause the I2C bus to hang if used, not much of an issue though
External pull-up resistorsBest option, but requires additional components

UART Throughput
The main UART function seems to be really slow, maxing out at about 250,000 bps throughput when set to 1,000,000 baud due to a gap of about 30us after each byte transmitted. This isn’t much of a problem at lower bauds, but as the baud rate increases it creates a huge overhead (75% @ 1Mbaud!). Attempting to receive data without a sufficient gap will also corrupt the data. Here are some screen shots comparing the MCP2221 and CH340 UART converter ICs transmitting data at 1,000,000 baud. The MCP2221 has a gap of about 30us between each byte, while the CH340 has a gap of just 1us.

MCP2221 @ 1Mbaud
CH340 @ 1Mbaud

Other issues
If the MCP2221 receives too much UART data too soon after powering up it will have trouble enumerating with the USB host.

MCP2221A
The MCP2221A replaces the MCP2221 and is fully compatible with this library. The new chip appears to have fixed the USB enumeration issue described above and the gap between each transmitted UART byte now varies from 4us to 18us, but usually around 6us.


A breakout module for the MCP2221,
design available on the GitHub repo