smhk

Calling MPSSE over FTDI from Python on CentOS 7

If wanting to use MPSSE commands with an FTDI device, you cannot simply use Python pyserial’s serial.Serial(). Instead you must use the FTDI driver in order to switch the device into MPSSE mode.

First the FTDI driver libftdi (and its dependency libusbx) need to be installed on CentOS 7:

yum install libusbx libftdi

Next the udev rules need to be configured to set the correct MODE for the FTDI device.

/etc/udev/rules.d/11-ftdi.rules §
SUBSYSTEM=="usb", ATTR{idVendor}=="0403", ATTR{idProduct}=="6001", MODE="0666"
SUBSYSTEM=="usb", ATTR{idVendor}=="0403", ATTR{idProduct}=="6011", MODE="0666"
SUBSYSTEM=="usb", ATTR{idVendor}=="0403", ATTR{idProduct}=="6010", MODE="0666"
SUBSYSTEM=="usb", ATTR{idVendor}=="0403", ATTR{idProduct}=="6014", MODE="0666"
SUBSYSTEM=="usb", ATTR{idVendor}=="0403", ATTR{idProduct}=="6015", MODE="0666"

(Optional: use the GROUP attribute to permit only certain users access).

Install pylibftdi, which allows calling FTDI methods from the libftdi driver.

Finally, you can write Python code to send MPSSE commands:

from pylibftdi import Device

BITMODE_MPSSE = 0x02
INTERFACE_A = 1

DIRECTION_MASK = 0x00

with dev as Device(interface_select=INTERFACE_A):
    dev.baudrate = 9600

    # This is the crucial part!!
    dev.ftdi_fn.ftdi_set_bitmode(DIRECTION_MASK, BITMODE_MPSSE)

    dev.write('\xDE\xAD\xBE\xEF')

The values for BITMODE_MPSSE and INTERFACE_A come from the FTDI documentation. Note that you will want to use either INTERFACE_A through INTERFACE_D accordingly depending upon your situation. Different FTDI chips have different numbers of interfaces.

Issues §

I found that testing this in the Python REPL worked fine, but upon integrating it with a larger code base Python would die with “Segmentation fault” upon the call to Device(). As a workaround, I found using lazy_open to delay opening the port along with disabling auto_detach to disable automatically closing the port fixed the issue. For example:

dev = Device(
    interface_select=INTERFACE_A,
    lazy_open=True,
    auto_detach=False,
)
dev.open()
# Do stuff
dev.close()

Alternatives §

pyftdi is more stable and contains more features than pylibftdi, however as of v0.22.1 it is Python 3 only, whereas pylibftdi supports Python 2 and 3.

There also exists libmpsse and FTD2XX which I have not tried.