src/jtag: add ftdi/libusb driver support for windows
Change-Id: Ibadb41463bbe16c453ab6de19f49643e18514e28
This commit is contained in:
parent
2c2e861b34
commit
aa75e393d2
33
configure.ac
33
configure.ac
|
@ -406,6 +406,21 @@ AC_ARG_ENABLE([remote-bitbang],
|
||||||
AS_HELP_STRING([--enable-remote-bitbang], [Enable building support for the Remote Bitbang driver]),
|
AS_HELP_STRING([--enable-remote-bitbang], [Enable building support for the Remote Bitbang driver]),
|
||||||
[build_remote_bitbang=$enableval], [build_remote_bitbang=yes])
|
[build_remote_bitbang=$enableval], [build_remote_bitbang=yes])
|
||||||
|
|
||||||
|
AC_MSG_CHECKING([whether to enable ftd2xx driver])
|
||||||
|
AS_CASE([$host_os],
|
||||||
|
[cygwin*|mingw*], [
|
||||||
|
AC_ARG_ENABLE([ftd2xx],
|
||||||
|
AS_HELP_STRING([--disable-ftd2xx], [Enable building support for native FTDI D2xx JTAG interface support (without libusb) on Windows.]),
|
||||||
|
[build_ftd2xx=$enableval], [build_ftd2xx=yes])
|
||||||
|
|
||||||
|
LDFLAGS="$LDFLAGS -l:ftd2xx.lib"
|
||||||
|
],
|
||||||
|
[
|
||||||
|
AS_IF([test "x$build_ftd2xx" = "xyes"], [
|
||||||
|
AC_MSG_ERROR([ftd2xx is only available on Windows])
|
||||||
|
])
|
||||||
|
])
|
||||||
|
|
||||||
AS_CASE(["${host_cpu}"],
|
AS_CASE(["${host_cpu}"],
|
||||||
[i?86|x86*], [],
|
[i?86|x86*], [],
|
||||||
[
|
[
|
||||||
|
@ -633,6 +648,19 @@ AS_IF([test "x$build_sysfsgpio" = "xyes"], [
|
||||||
AC_DEFINE([BUILD_SYSFSGPIO], [0], [0 if you don't want SysfsGPIO driver.])
|
AC_DEFINE([BUILD_SYSFSGPIO], [0], [0 if you don't want SysfsGPIO driver.])
|
||||||
])
|
])
|
||||||
|
|
||||||
|
AS_IF([test "x$build_xlnx_pcie_xvc" = "xyes"], [
|
||||||
|
build_xlnx_pcie_xvc=yes
|
||||||
|
AC_DEFINE([BUILD_XLNX_PCIE_XVC], [1], [1 if you want the Xilinx XVC/PCIe driver.])
|
||||||
|
], [
|
||||||
|
AC_DEFINE([BUILD_XLNX_PCIE_XVC], [0], [0 if you don't want Xilinx XVC/PCIe driver.])
|
||||||
|
])
|
||||||
|
|
||||||
|
AS_IF([test "x$build_ftd2xx" = "xyes"], [
|
||||||
|
AC_DEFINE([BUILD_FTD2XX], [1], [1 if you want the FTD2xx driver support.])
|
||||||
|
], [
|
||||||
|
AC_DEFINE([BUILD_FTD2XX], [0], [0 if you don't want FTD2xx driver support.])
|
||||||
|
])
|
||||||
|
|
||||||
PKG_CHECK_MODULES([LIBUSB1], [libusb-1.0], [
|
PKG_CHECK_MODULES([LIBUSB1], [libusb-1.0], [
|
||||||
use_libusb1=yes
|
use_libusb1=yes
|
||||||
AC_DEFINE([HAVE_LIBUSB1], [1], [Define if you have libusb-1.x])
|
AC_DEFINE([HAVE_LIBUSB1], [1], [Define if you have libusb-1.x])
|
||||||
|
@ -796,6 +824,7 @@ AM_CONDITIONAL([USE_LIBJAYLINK], [test "x$use_libjaylink" = "xyes"])
|
||||||
AM_CONDITIONAL([RSHIM], [test "x$build_rshim" = "xyes"])
|
AM_CONDITIONAL([RSHIM], [test "x$build_rshim" = "xyes"])
|
||||||
AM_CONDITIONAL([DMEM], [test "x$build_dmem" = "xyes"])
|
AM_CONDITIONAL([DMEM], [test "x$build_dmem" = "xyes"])
|
||||||
AM_CONDITIONAL([HAVE_CAPSTONE], [test "x$enable_capstone" != "xno"])
|
AM_CONDITIONAL([HAVE_CAPSTONE], [test "x$enable_capstone" != "xno"])
|
||||||
|
AM_CONDITIONAL([USE_FTD2XX], [test "x$build_ftd2xx" = "xyes"])
|
||||||
|
|
||||||
AM_CONDITIONAL([INTERNAL_JIMTCL], [test "x$use_internal_jimtcl" = "xyes"])
|
AM_CONDITIONAL([INTERNAL_JIMTCL], [test "x$use_internal_jimtcl" = "xyes"])
|
||||||
AM_CONDITIONAL([HAVE_JIMTCL_PKG_CONFIG], [test "x$have_jimtcl_pkg_config" = "xyes"])
|
AM_CONDITIONAL([HAVE_JIMTCL_PKG_CONFIG], [test "x$have_jimtcl_pkg_config" = "xyes"])
|
||||||
|
@ -898,3 +927,7 @@ m4_foreach([adapter], [USB1_ADAPTERS,
|
||||||
])
|
])
|
||||||
])
|
])
|
||||||
echo
|
echo
|
||||||
|
|
||||||
|
AS_IF([test "x$build_ftd2xx" = "xyes"], [
|
||||||
|
echo "FTD2xx driver support enabled."
|
||||||
|
])
|
||||||
|
|
|
@ -14,6 +14,9 @@
|
||||||
#ifndef OPENOCD_HELPER_REPLACEMENTS_H
|
#ifndef OPENOCD_HELPER_REPLACEMENTS_H
|
||||||
#define OPENOCD_HELPER_REPLACEMENTS_H
|
#define OPENOCD_HELPER_REPLACEMENTS_H
|
||||||
|
|
||||||
|
#define HAVE_GETTIMEOFDAY 1
|
||||||
|
#define HAVE_USLEEP 1
|
||||||
|
|
||||||
#include <stdint.h>
|
#include <stdint.h>
|
||||||
#include <helper/system.h>
|
#include <helper/system.h>
|
||||||
|
|
||||||
|
|
|
@ -0,0 +1,87 @@
|
||||||
|
#ifndef __WINDOWS_TYPES__
|
||||||
|
#define __WINDOWS_TYPES__
|
||||||
|
|
||||||
|
#define MAX_NUM_DEVICES 50
|
||||||
|
#include <sys/time.h>
|
||||||
|
|
||||||
|
typedef unsigned long DWORD;
|
||||||
|
typedef unsigned long ULONG;
|
||||||
|
typedef unsigned short USHORT;
|
||||||
|
typedef short SHORT;
|
||||||
|
typedef unsigned char UCHAR;
|
||||||
|
typedef unsigned short WORD;
|
||||||
|
typedef unsigned char BYTE;
|
||||||
|
typedef unsigned char *LPBYTE;
|
||||||
|
typedef int BOOL;
|
||||||
|
typedef char BOOLEAN;
|
||||||
|
typedef char CHAR;
|
||||||
|
typedef int *LPBOOL;
|
||||||
|
typedef unsigned char *PUCHAR;
|
||||||
|
typedef const char *LPCSTR;
|
||||||
|
typedef char *PCHAR;
|
||||||
|
typedef void *PVOID;
|
||||||
|
typedef void *HANDLE;
|
||||||
|
typedef long LONG;
|
||||||
|
typedef int INT;
|
||||||
|
typedef unsigned int UINT;
|
||||||
|
typedef char *LPSTR;
|
||||||
|
typedef char *LPTSTR;
|
||||||
|
typedef DWORD *LPDWORD;
|
||||||
|
typedef WORD *LPWORD;
|
||||||
|
typedef ULONG *PULONG;
|
||||||
|
typedef PVOID LPVOID;
|
||||||
|
typedef void VOID;
|
||||||
|
typedef unsigned long long int ULONGLONG;
|
||||||
|
|
||||||
|
typedef struct _OVERLAPPED {
|
||||||
|
DWORD Internal;
|
||||||
|
DWORD InternalHigh;
|
||||||
|
DWORD Offset;
|
||||||
|
DWORD OffsetHigh;
|
||||||
|
HANDLE hEvent;
|
||||||
|
} OVERLAPPED, *LPOVERLAPPED;
|
||||||
|
|
||||||
|
typedef struct _SECURITY_ATTRIBUTES {
|
||||||
|
DWORD nLength;
|
||||||
|
LPVOID lpSecurityDescriptor;
|
||||||
|
BOOL bInheritHandle;
|
||||||
|
} SECURITY_ATTRIBUTES , *LPSECURITY_ATTRIBUTES;
|
||||||
|
|
||||||
|
typedef struct timeval SYSTEMTIME;
|
||||||
|
typedef struct timeval FILETIME;
|
||||||
|
#ifndef TRUE
|
||||||
|
#define TRUE 1
|
||||||
|
#endif
|
||||||
|
#ifndef FALSE
|
||||||
|
#define FALSE 0
|
||||||
|
#endif
|
||||||
|
|
||||||
|
//
|
||||||
|
// Modem Status Flags
|
||||||
|
//
|
||||||
|
#define MS_CTS_ON ((DWORD)0x0010)
|
||||||
|
#define MS_DSR_ON ((DWORD)0x0020)
|
||||||
|
#define MS_RING_ON ((DWORD)0x0040)
|
||||||
|
#define MS_RLSD_ON ((DWORD)0x0080)
|
||||||
|
|
||||||
|
//
|
||||||
|
// Error Flags
|
||||||
|
//
|
||||||
|
|
||||||
|
#define CE_RXOVER 0x0001 // Receive Queue overflow
|
||||||
|
#define CE_OVERRUN 0x0002 // Receive Overrun Error
|
||||||
|
#define CE_RXPARITY 0x0004 // Receive Parity Error
|
||||||
|
#define CE_FRAME 0x0008 // Receive Framing error
|
||||||
|
#define CE_BREAK 0x0010 // Break Detected
|
||||||
|
#define CE_TXFULL 0x0100 // TX Queue is full
|
||||||
|
#define CE_PTO 0x0200 // LPTx Timeout
|
||||||
|
#define CE_IOE 0x0400 // LPTx I/O Error
|
||||||
|
#define CE_DNS 0x0800 // LPTx Device not selected
|
||||||
|
#define CE_OOP 0x1000 // LPTx Out-Of-Paper
|
||||||
|
#define CE_MODE 0x8000 // Requested mode unsupported
|
||||||
|
|
||||||
|
#ifndef INVALID_HANDLE_VALUE
|
||||||
|
#define INVALID_HANDLE_VALUE 0xFFFFFFFF
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#endif
|
File diff suppressed because it is too large
Load Diff
|
@ -16,6 +16,29 @@
|
||||||
#include "libusb_helper.h"
|
#include "libusb_helper.h"
|
||||||
#include <libusb.h>
|
#include <libusb.h>
|
||||||
|
|
||||||
|
#if defined(_WIN32) && BUILD_FTD2XX == 1
|
||||||
|
#define BUILD_BACKEND_FTD2XX
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#ifdef BUILD_BACKEND_FTD2XX
|
||||||
|
#include "ftd2xx/ftd2xx.h"
|
||||||
|
|
||||||
|
#define FTD2XX_CHANNEL_MIN 0
|
||||||
|
#define FTD2XX_CHANNEL_MAX 3
|
||||||
|
char *ftd2xx_channel_names[] = {
|
||||||
|
"A",
|
||||||
|
"B",
|
||||||
|
"C",
|
||||||
|
"D",
|
||||||
|
};
|
||||||
|
#endif // BUILD_BACKEND_FTD2XX
|
||||||
|
|
||||||
|
// FTD2XX and libusb compatibility
|
||||||
|
#define BACKEND_DIVERGENCE_START switch (ctx->backend) { default:;
|
||||||
|
#define BACKEND_DIVERGENCE_LIBUSB break; case MPSSE_BACKEND_TYPE_LIBUSB:;
|
||||||
|
#define BACKEND_DIVERGENCE_FTD2XX break; case MPSSE_BACKEND_TYPE_FTD2XX:;
|
||||||
|
#define BACKEND_DIVERGENCE_END break; }
|
||||||
|
|
||||||
/* Compatibility define for older libusb-1.0 */
|
/* Compatibility define for older libusb-1.0 */
|
||||||
#ifndef LIBUSB_CALL
|
#ifndef LIBUSB_CALL
|
||||||
#define LIBUSB_CALL
|
#define LIBUSB_CALL
|
||||||
|
@ -53,22 +76,30 @@
|
||||||
#define SIO_RESET_PURGE_TX 2
|
#define SIO_RESET_PURGE_TX 2
|
||||||
|
|
||||||
struct mpsse_ctx {
|
struct mpsse_ctx {
|
||||||
|
enum mpsse_backend_type backend;
|
||||||
|
|
||||||
|
#ifdef BUILD_BACKEND_FTD2XX
|
||||||
|
// ftd2xx backend
|
||||||
|
FT_HANDLE usb_ft_handle;
|
||||||
|
#endif
|
||||||
|
// libusb backend
|
||||||
struct libusb_context *usb_ctx;
|
struct libusb_context *usb_ctx;
|
||||||
struct libusb_device_handle *usb_dev;
|
struct libusb_device_handle *usb_dev;
|
||||||
|
|
||||||
unsigned int usb_write_timeout;
|
unsigned int usb_write_timeout;
|
||||||
unsigned int usb_read_timeout;
|
unsigned int usb_read_timeout;
|
||||||
uint8_t in_ep;
|
uint8_t in_ep;
|
||||||
uint8_t out_ep;
|
uint8_t out_ep;
|
||||||
uint16_t max_packet_size;
|
uint16_t max_packet_size;
|
||||||
uint16_t index;
|
uint16_t index;
|
||||||
uint8_t interface;
|
uint8_t interface; // channel
|
||||||
enum ftdi_chip_type type;
|
enum ftdi_chip_type type;
|
||||||
uint8_t *write_buffer;
|
uint8_t *write_buffer;
|
||||||
unsigned int write_size;
|
unsigned write_size; // size of write_buffer
|
||||||
unsigned int write_count;
|
unsigned write_count; // size of data being written
|
||||||
uint8_t *read_buffer;
|
uint8_t *read_buffer;
|
||||||
unsigned int read_size;
|
unsigned read_size; // size of read_buffer
|
||||||
unsigned int read_count;
|
unsigned read_count; // size of data needed to be read
|
||||||
uint8_t *read_chunk;
|
uint8_t *read_chunk;
|
||||||
unsigned int read_chunk_size;
|
unsigned int read_chunk_size;
|
||||||
struct bit_copy_queue read_queue;
|
struct bit_copy_queue read_queue;
|
||||||
|
@ -159,6 +190,8 @@ static bool open_matching_device(struct mpsse_ctx *ctx, const uint16_t vids[], c
|
||||||
struct libusb_config_descriptor *config0;
|
struct libusb_config_descriptor *config0;
|
||||||
int err;
|
int err;
|
||||||
bool found = false;
|
bool found = false;
|
||||||
|
|
||||||
|
// try libusb first
|
||||||
ssize_t cnt = libusb_get_device_list(ctx->usb_ctx, &list);
|
ssize_t cnt = libusb_get_device_list(ctx->usb_ctx, &list);
|
||||||
if (cnt < 0)
|
if (cnt < 0)
|
||||||
LOG_ERROR("libusb_get_device_list() failed with %s", libusb_error_name(cnt));
|
LOG_ERROR("libusb_get_device_list() failed with %s", libusb_error_name(cnt));
|
||||||
|
@ -177,7 +210,7 @@ static bool open_matching_device(struct mpsse_ctx *ctx, const uint16_t vids[], c
|
||||||
|
|
||||||
err = libusb_open(device, &ctx->usb_dev);
|
err = libusb_open(device, &ctx->usb_dev);
|
||||||
if (err != LIBUSB_SUCCESS) {
|
if (err != LIBUSB_SUCCESS) {
|
||||||
LOG_ERROR("libusb_open() failed with %s",
|
LOG_INFO("libusb_open() failed with %s",
|
||||||
libusb_error_name(err));
|
libusb_error_name(err));
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
@ -204,10 +237,14 @@ static bool open_matching_device(struct mpsse_ctx *ctx, const uint16_t vids[], c
|
||||||
libusb_free_device_list(list, 1);
|
libusb_free_device_list(list, 1);
|
||||||
|
|
||||||
if (!found) {
|
if (!found) {
|
||||||
/* The caller reports detailed error desc */
|
LOG_INFO("no device found, trying D2xx driver");
|
||||||
return false;
|
goto libusb_abort;
|
||||||
|
} else {
|
||||||
|
LOG_INFO("Using libusb driver");
|
||||||
}
|
}
|
||||||
|
|
||||||
|
ctx -> backend = MPSSE_BACKEND_TYPE_LIBUSB;
|
||||||
|
|
||||||
err = libusb_get_config_descriptor(libusb_get_device(ctx->usb_dev), 0, &config0);
|
err = libusb_get_config_descriptor(libusb_get_device(ctx->usb_dev), 0, &config0);
|
||||||
if (err != LIBUSB_SUCCESS) {
|
if (err != LIBUSB_SUCCESS) {
|
||||||
LOG_ERROR("libusb_get_config_descriptor() failed with %s", libusb_error_name(err));
|
LOG_ERROR("libusb_get_config_descriptor() failed with %s", libusb_error_name(err));
|
||||||
|
@ -289,7 +326,7 @@ static bool open_matching_device(struct mpsse_ctx *ctx, const uint16_t vids[], c
|
||||||
ctx->type = TYPE_FT4232HA;
|
ctx->type = TYPE_FT4232HA;
|
||||||
break;
|
break;
|
||||||
default:
|
default:
|
||||||
LOG_ERROR("unsupported FTDI chip type: 0x%04x", desc.bcdDevice);
|
LOG_ERROR("unsupported FTDI chip type (libusb): 0x%04x", desc.bcdDevice);
|
||||||
goto error;
|
goto error;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -321,11 +358,151 @@ static bool open_matching_device(struct mpsse_ctx *ctx, const uint16_t vids[], c
|
||||||
libusb_free_config_descriptor(config0);
|
libusb_free_config_descriptor(config0);
|
||||||
return true;
|
return true;
|
||||||
|
|
||||||
|
libusb_abort:
|
||||||
|
#ifdef BUILD_BACKEND_FTD2XX
|
||||||
|
// try FTD2XX
|
||||||
|
|
||||||
|
LOG_DEBUG("open_matching_device() FTD2xx init start");
|
||||||
|
FT_STATUS ft_status;
|
||||||
|
DWORD ft_cnt;
|
||||||
|
ft_status = FT_CreateDeviceInfoList(&ft_cnt);
|
||||||
|
if (ft_status != FT_OK) {
|
||||||
|
LOG_ERROR("FT_CreateDeviceInfoList() error %lu", ft_status);
|
||||||
|
goto open_matching_device_fallback_libusb;
|
||||||
|
}
|
||||||
|
LOG_INFO("D2xx device count: %lu", ft_cnt);
|
||||||
|
|
||||||
|
// FTD2xx (except single-channel chips like FT232H) descriptor is "USB <-> JTAG-DEBUGGER A"
|
||||||
|
// libusb descriptor is "USB <-> JTAG-DEBUGGER"
|
||||||
|
// so the suffix is mapped to libusb channels for compatibility
|
||||||
|
char product_with_suffix[256];
|
||||||
|
if (product && ctx->interface >= FTD2XX_CHANNEL_MIN && ctx->interface <= FTD2XX_CHANNEL_MAX) {
|
||||||
|
snprintf(product_with_suffix, sizeof(product_with_suffix), "%s %s", product, ftd2xx_channel_names[ctx->interface]);
|
||||||
|
} else {
|
||||||
|
snprintf(product_with_suffix, sizeof(product_with_suffix), "%s", product);
|
||||||
|
}
|
||||||
|
|
||||||
|
FT_DEVICE_LIST_INFO_NODE *devInfo;
|
||||||
|
devInfo = (FT_DEVICE_LIST_INFO_NODE*)malloc(sizeof(FT_DEVICE_LIST_INFO_NODE) * ft_cnt);
|
||||||
|
DWORD ft_matched_device_id;
|
||||||
|
char *ft_matched_device_description = NULL;
|
||||||
|
if (ft_cnt > 0) {
|
||||||
|
ft_status = FT_GetDeviceInfoList(devInfo, &ft_cnt);
|
||||||
|
if (ft_status == FT_OK) {
|
||||||
|
for (int i = 0; i < ft_cnt; i++) {
|
||||||
|
DWORD device_vid = devInfo[i].ID >> 16; // higher 16 bits
|
||||||
|
DWORD device_pid = devInfo[i].ID & 65535; // lower 16 bits
|
||||||
|
|
||||||
|
// LOG_DEBUG("FTD2xx Device #%d:", i);
|
||||||
|
// LOG_DEBUG(" Flags=0x%lx", devInfo[i].Flags);
|
||||||
|
// LOG_DEBUG(" Type=0x%lx", devInfo[i].Type);
|
||||||
|
// LOG_DEBUG(" ID=0x%lx (VID=0x%04lx, PID=0x%04lx)", devInfo[i].ID, device_vid, device_pid);
|
||||||
|
// LOG_DEBUG(" LocId=0x%lx", devInfo[i].LocId);
|
||||||
|
// LOG_DEBUG(" SerialNumber=%s", devInfo[i].SerialNumber);
|
||||||
|
// LOG_DEBUG(" Description=%s", devInfo[i].Description);
|
||||||
|
// LOG_DEBUG(" ftHandle=0x%p", devInfo[i].ftHandle);
|
||||||
|
|
||||||
|
if (vids && *vids != device_vid) continue;
|
||||||
|
if (pids && *pids != device_pid) continue;
|
||||||
|
|
||||||
|
// FIXME: check location does not work for now since libusb use a different location identifier from FTD2xx
|
||||||
|
|
||||||
|
if (product) {
|
||||||
|
if (!strcmp(devInfo[i].Description, product)) { // whole string match -- for FT232H with only one channel
|
||||||
|
ft_matched_device_description = product;
|
||||||
|
}
|
||||||
|
else if (!strcmp(devInfo[i].Description, product_with_suffix)) { // for FT2232H, etc. with multiple channels
|
||||||
|
ft_matched_device_description = product_with_suffix;
|
||||||
|
} else {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (serial && strcmp(devInfo[i].SerialNumber, serial)) continue;
|
||||||
|
|
||||||
|
found = true;
|
||||||
|
ft_matched_device_id = i;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
LOG_ERROR("FT_GetDeviceInfoList() error %lu", ft_status);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!found) {
|
||||||
|
LOG_WARNING("D2xx driver found nothing, falling back to libusb...");
|
||||||
|
goto open_matching_device_fallback_libusb;
|
||||||
|
} else {
|
||||||
|
LOG_INFO("Connecting to \"%s\" using D2xx mode...", ft_matched_device_description);
|
||||||
|
}
|
||||||
|
|
||||||
|
ctx -> backend = MPSSE_BACKEND_TYPE_FTD2XX;
|
||||||
|
|
||||||
|
switch (devInfo[ft_matched_device_id].Type) {
|
||||||
|
case FT_DEVICE_2232C:
|
||||||
|
ctx->type = TYPE_FT2232C;
|
||||||
|
break;
|
||||||
|
case FT_DEVICE_2232H:
|
||||||
|
ctx->type = TYPE_FT2232H;
|
||||||
|
break;
|
||||||
|
case FT_DEVICE_4232H:
|
||||||
|
ctx->type = TYPE_FT4232H;
|
||||||
|
break;
|
||||||
|
case FT_DEVICE_232H:
|
||||||
|
ctx->type = TYPE_FT232H;
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
LOG_ERROR("unsupported FTDI chip type (D2xx): 0x%04x", devInfo[ft_matched_device_id].Type);
|
||||||
|
goto error;
|
||||||
|
}
|
||||||
|
|
||||||
|
ft_status = FT_Open(ft_matched_device_id, &(ctx->usb_ft_handle));
|
||||||
|
if (ft_status != FT_OK)
|
||||||
|
{
|
||||||
|
LOG_ERROR("FT_Open() Failed with error %lu\n", ft_status);
|
||||||
|
goto error;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Reset USB device
|
||||||
|
ft_status |= FT_ResetDevice(ctx->usb_ft_handle);
|
||||||
|
|
||||||
|
// Set parameters
|
||||||
|
ft_status |= FT_SetUSBParameters(ctx->usb_ft_handle, ctx->max_packet_size, ctx->max_packet_size); // Set USB request transfer sizes
|
||||||
|
ft_status |= FT_SetChars(ctx->usb_ft_handle, false, 0, false, 0); // Disable event and error characters
|
||||||
|
// ft_status |= FT_SetFlowControl(ctx->usb_ft_handle, FT_FLOW_RTS_CTS, 0x00, 0x00); // Turn on flow control to synchronize IN requests
|
||||||
|
// ft_status |= FT_SetBaudRate(ctx->usb_ft_handle, 1000000);
|
||||||
|
|
||||||
|
// NOTE: this assumes usb timeouts does not change across the lifetime of the process;
|
||||||
|
// on the libusb side, timeouts are set per request.
|
||||||
|
ft_status |= FT_SetTimeouts(ctx->usb_ft_handle, ctx->usb_read_timeout, ctx->usb_write_timeout); // Sets the read and write timeouts in milliseconds
|
||||||
|
|
||||||
|
if (ft_status != FT_OK)
|
||||||
|
{
|
||||||
|
LOG_ERROR("Error in initializing the MPSSE %lu\n", ft_status);
|
||||||
|
FT_Close(ctx->usb_ft_handle);
|
||||||
|
goto open_matching_device_fallback_libusb;
|
||||||
|
}
|
||||||
|
|
||||||
|
return true;
|
||||||
|
|
||||||
|
open_matching_device_fallback_libusb:
|
||||||
|
LOG_DEBUG("open_matching_device() FTD2xx init end");
|
||||||
|
#endif // BUILD_BACKEND_FTD2XX
|
||||||
|
|
||||||
desc_error:
|
desc_error:
|
||||||
LOG_ERROR("unrecognized USB device descriptor");
|
LOG_ERROR("unrecognized USB device descriptor");
|
||||||
error:
|
error:
|
||||||
|
BACKEND_DIVERGENCE_START
|
||||||
|
#ifdef BUILD_BACKEND_FTD2XX
|
||||||
|
BACKEND_DIVERGENCE_FTD2XX
|
||||||
|
FT_SetBitMode(ctx->usb_ft_handle, 0x0, 0x00);
|
||||||
|
FT_Close(ctx->usb_ft_handle);
|
||||||
|
#endif // BUILD_BACKEND_FTD2XX
|
||||||
|
BACKEND_DIVERGENCE_LIBUSB
|
||||||
libusb_free_config_descriptor(config0);
|
libusb_free_config_descriptor(config0);
|
||||||
libusb_close(ctx->usb_dev);
|
libusb_close(ctx->usb_dev);
|
||||||
|
BACKEND_DIVERGENCE_END
|
||||||
|
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -334,6 +511,9 @@ struct mpsse_ctx *mpsse_open(const uint16_t vids[], const uint16_t pids[], const
|
||||||
{
|
{
|
||||||
struct mpsse_ctx *ctx = calloc(1, sizeof(*ctx));
|
struct mpsse_ctx *ctx = calloc(1, sizeof(*ctx));
|
||||||
int err;
|
int err;
|
||||||
|
#ifdef BUILD_BACKEND_FTD2XX
|
||||||
|
FT_STATUS ft_status;
|
||||||
|
#endif // BUILD_BACKEND_FTD2XX
|
||||||
|
|
||||||
if (!ctx)
|
if (!ctx)
|
||||||
return NULL;
|
return NULL;
|
||||||
|
@ -375,6 +555,9 @@ struct mpsse_ctx *mpsse_open(const uint16_t vids[], const uint16_t pids[], const
|
||||||
goto error;
|
goto error;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Set the latency timer (default is 16ms)
|
||||||
|
BACKEND_DIVERGENCE_START
|
||||||
|
BACKEND_DIVERGENCE_LIBUSB
|
||||||
err = libusb_control_transfer(ctx->usb_dev, FTDI_DEVICE_OUT_REQTYPE,
|
err = libusb_control_transfer(ctx->usb_dev, FTDI_DEVICE_OUT_REQTYPE,
|
||||||
SIO_SET_LATENCY_TIMER_REQUEST, 255, ctx->index, NULL, 0,
|
SIO_SET_LATENCY_TIMER_REQUEST, 255, ctx->index, NULL, 0,
|
||||||
ctx->usb_write_timeout);
|
ctx->usb_write_timeout);
|
||||||
|
@ -382,7 +565,19 @@ struct mpsse_ctx *mpsse_open(const uint16_t vids[], const uint16_t pids[], const
|
||||||
LOG_ERROR("unable to set latency timer: %s", libusb_error_name(err));
|
LOG_ERROR("unable to set latency timer: %s", libusb_error_name(err));
|
||||||
goto error;
|
goto error;
|
||||||
}
|
}
|
||||||
|
#ifdef BUILD_BACKEND_FTD2XX
|
||||||
|
BACKEND_DIVERGENCE_FTD2XX
|
||||||
|
ft_status = FT_SetLatencyTimer(ctx->usb_ft_handle, 255);
|
||||||
|
if (ft_status != FT_OK) {
|
||||||
|
LOG_ERROR("unable to set latency timer: %lu", ft_status);
|
||||||
|
goto error;
|
||||||
|
}
|
||||||
|
#endif // BUILD_BACKEND_FTD2XX
|
||||||
|
BACKEND_DIVERGENCE_END
|
||||||
|
|
||||||
|
// set MPSSE bitmode
|
||||||
|
BACKEND_DIVERGENCE_START
|
||||||
|
BACKEND_DIVERGENCE_LIBUSB
|
||||||
err = libusb_control_transfer(ctx->usb_dev,
|
err = libusb_control_transfer(ctx->usb_dev,
|
||||||
FTDI_DEVICE_OUT_REQTYPE,
|
FTDI_DEVICE_OUT_REQTYPE,
|
||||||
SIO_SET_BITMODE_REQUEST,
|
SIO_SET_BITMODE_REQUEST,
|
||||||
|
@ -395,6 +590,19 @@ struct mpsse_ctx *mpsse_open(const uint16_t vids[], const uint16_t pids[], const
|
||||||
LOG_ERROR("unable to set MPSSE bitmode: %s", libusb_error_name(err));
|
LOG_ERROR("unable to set MPSSE bitmode: %s", libusb_error_name(err));
|
||||||
goto error;
|
goto error;
|
||||||
}
|
}
|
||||||
|
#ifdef BUILD_BACKEND_FTD2XX
|
||||||
|
BACKEND_DIVERGENCE_FTD2XX
|
||||||
|
ft_status |= FT_SetBitMode(ctx->usb_ft_handle, 0x0, 0x00); // reset controller mode
|
||||||
|
ft_status |= FT_SetBitMode(ctx->usb_ft_handle, 0x0, BITMODE_MPSSE); // enter MPSSE mode
|
||||||
|
if (ft_status != FT_OK) {
|
||||||
|
LOG_ERROR("unable to set MPSSE bitmode: %lu", ft_status);
|
||||||
|
goto error;
|
||||||
|
}
|
||||||
|
|
||||||
|
// documentation (https://www.ftdichip.com/Support/Documents/AppNotes/AN_135_MPSSE_Basics.pdf) said to hardcode a delay here
|
||||||
|
Sleep(50);
|
||||||
|
#endif // BUILD_BACKEND_FTD2XX
|
||||||
|
BACKEND_DIVERGENCE_END
|
||||||
|
|
||||||
mpsse_purge(ctx);
|
mpsse_purge(ctx);
|
||||||
|
|
||||||
|
@ -406,10 +614,19 @@ error:
|
||||||
|
|
||||||
void mpsse_close(struct mpsse_ctx *ctx)
|
void mpsse_close(struct mpsse_ctx *ctx)
|
||||||
{
|
{
|
||||||
|
BACKEND_DIVERGENCE_START
|
||||||
|
BACKEND_DIVERGENCE_LIBUSB
|
||||||
if (ctx->usb_dev)
|
if (ctx->usb_dev)
|
||||||
libusb_close(ctx->usb_dev);
|
libusb_close(ctx->usb_dev);
|
||||||
if (ctx->usb_ctx)
|
if (ctx->usb_ctx)
|
||||||
libusb_exit(ctx->usb_ctx);
|
libusb_exit(ctx->usb_ctx);
|
||||||
|
#ifdef BUILD_BACKEND_FTD2XX
|
||||||
|
BACKEND_DIVERGENCE_FTD2XX
|
||||||
|
FT_SetBitMode(ctx->usb_ft_handle, 0x0, 0x00); // reset controller mode
|
||||||
|
FT_Close(ctx->usb_ft_handle);
|
||||||
|
#endif // BUILD_BACKEND_FTD2XX
|
||||||
|
BACKEND_DIVERGENCE_END
|
||||||
|
|
||||||
bit_copy_discard(&ctx->read_queue);
|
bit_copy_discard(&ctx->read_queue);
|
||||||
|
|
||||||
free(ctx->write_buffer);
|
free(ctx->write_buffer);
|
||||||
|
@ -431,6 +648,10 @@ static void mpsse_purge(struct mpsse_ctx *ctx)
|
||||||
ctx->read_count = 0;
|
ctx->read_count = 0;
|
||||||
ctx->retval = ERROR_OK;
|
ctx->retval = ERROR_OK;
|
||||||
bit_copy_discard(&ctx->read_queue);
|
bit_copy_discard(&ctx->read_queue);
|
||||||
|
|
||||||
|
// purge RX & TX buffer
|
||||||
|
BACKEND_DIVERGENCE_START
|
||||||
|
BACKEND_DIVERGENCE_LIBUSB
|
||||||
err = libusb_control_transfer(ctx->usb_dev, FTDI_DEVICE_OUT_REQTYPE, SIO_RESET_REQUEST,
|
err = libusb_control_transfer(ctx->usb_dev, FTDI_DEVICE_OUT_REQTYPE, SIO_RESET_REQUEST,
|
||||||
SIO_RESET_PURGE_RX, ctx->index, NULL, 0, ctx->usb_write_timeout);
|
SIO_RESET_PURGE_RX, ctx->index, NULL, 0, ctx->usb_write_timeout);
|
||||||
if (err < 0) {
|
if (err < 0) {
|
||||||
|
@ -444,6 +665,15 @@ static void mpsse_purge(struct mpsse_ctx *ctx)
|
||||||
LOG_ERROR("unable to purge ftdi tx buffers: %s", libusb_error_name(err));
|
LOG_ERROR("unable to purge ftdi tx buffers: %s", libusb_error_name(err));
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
#ifdef BUILD_BACKEND_FTD2XX
|
||||||
|
BACKEND_DIVERGENCE_FTD2XX
|
||||||
|
FT_STATUS ft_status = FT_Purge(ctx->usb_ft_handle, FT_PURGE_TX | FT_PURGE_RX);
|
||||||
|
if (ft_status != FT_OK) {
|
||||||
|
LOG_ERROR("unable to purge ftdi tx&rx buffers: %ul", ft_status);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
#endif // BUILD_BACKEND_FTD2XX
|
||||||
|
BACKEND_DIVERGENCE_END
|
||||||
}
|
}
|
||||||
|
|
||||||
static unsigned int buffer_write_space(struct mpsse_ctx *ctx)
|
static unsigned int buffer_write_space(struct mpsse_ctx *ctx)
|
||||||
|
@ -865,7 +1095,82 @@ int mpsse_flush(struct mpsse_ctx *ctx)
|
||||||
if (ctx->write_count == 0)
|
if (ctx->write_count == 0)
|
||||||
return retval;
|
return retval;
|
||||||
|
|
||||||
struct libusb_transfer *read_transfer = NULL;
|
BACKEND_DIVERGENCE_START
|
||||||
|
#ifdef BUILD_BACKEND_FTD2XX
|
||||||
|
BACKEND_DIVERGENCE_FTD2XX
|
||||||
|
|
||||||
|
FT_STATUS ft_status;
|
||||||
|
DWORD read_bytes_done, write_bytes_done;
|
||||||
|
|
||||||
|
if (ctx->read_count) {
|
||||||
|
buffer_write_byte(ctx, 0x87); /* SEND_IMMEDIATE */
|
||||||
|
}
|
||||||
|
|
||||||
|
DWORD rx_queue_len, tx_queue_len, ft_event_status; // temporary, for FT_GetStatus calls
|
||||||
|
|
||||||
|
// write
|
||||||
|
if (ctx->write_count) {
|
||||||
|
assert(ctx->write_count <= ctx->write_size); // assume write_buffer can be sent in one request
|
||||||
|
|
||||||
|
ft_status = FT_Write(ctx->usb_ft_handle, ctx->write_buffer, ctx->write_count, &write_bytes_done);
|
||||||
|
LOG_DEBUG_IO("FT_Write(): returned %lu, written bytes %lu/%u", ft_status, write_bytes_done, ctx->write_count);
|
||||||
|
|
||||||
|
// wait for queue to clear
|
||||||
|
do {
|
||||||
|
ft_status = FT_GetStatus(ctx->usb_ft_handle, &rx_queue_len, &tx_queue_len, &ft_event_status);
|
||||||
|
if (ft_status != FT_OK) {
|
||||||
|
LOG_ERROR("FT_GetStatus() returned %lu", ft_status);
|
||||||
|
}
|
||||||
|
keep_alive();
|
||||||
|
} while (tx_queue_len);
|
||||||
|
}
|
||||||
|
|
||||||
|
// read
|
||||||
|
if (ctx->read_count) {
|
||||||
|
assert(ctx->read_count <= ctx->read_size); // assume read_buffer can be read in one request
|
||||||
|
|
||||||
|
// read
|
||||||
|
// BTW, read_chunk* is not needed since FT_Read removes the 2 status bytes (reference: read_cb()) for us
|
||||||
|
ft_status = FT_Read(ctx->usb_ft_handle, ctx->read_buffer, ctx->read_count, &read_bytes_done);
|
||||||
|
LOG_DEBUG_IO("FT_Read(): returned %lu, read bytes %lu/%u", ft_status, read_bytes_done, ctx->read_count);
|
||||||
|
|
||||||
|
// wait for queue to clear
|
||||||
|
do {
|
||||||
|
ft_status = FT_GetStatus(ctx->usb_ft_handle, &rx_queue_len, &tx_queue_len, &ft_event_status);
|
||||||
|
if (ft_status != FT_OK) {
|
||||||
|
LOG_ERROR("FT_GetStatus() returned %lu", ft_status);
|
||||||
|
}
|
||||||
|
keep_alive();
|
||||||
|
} while (rx_queue_len);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (ft_status != FT_OK) {
|
||||||
|
// LOG_ERROR("FTD2xx failed with %lu", ft_status);
|
||||||
|
retval = ERROR_FAIL;
|
||||||
|
} else if (write_bytes_done < ctx->write_count) {
|
||||||
|
LOG_ERROR("ftdi device did not accept all data: %d, tried %d",
|
||||||
|
write_bytes_done,
|
||||||
|
ctx->write_count);
|
||||||
|
retval = ERROR_FAIL;
|
||||||
|
} else if (read_bytes_done < ctx->read_count) {
|
||||||
|
LOG_ERROR("ftdi device did not return all data: %d, expected %d",
|
||||||
|
read_bytes_done,
|
||||||
|
ctx->read_count);
|
||||||
|
retval = ERROR_FAIL;
|
||||||
|
} else if (ctx->read_count) {
|
||||||
|
ctx->write_count = 0;
|
||||||
|
ctx->read_count = 0;
|
||||||
|
bit_copy_execute(&ctx->read_queue);
|
||||||
|
retval = ERROR_OK;
|
||||||
|
} else {
|
||||||
|
ctx->write_count = 0;
|
||||||
|
bit_copy_discard(&ctx->read_queue);
|
||||||
|
retval = ERROR_OK;
|
||||||
|
}
|
||||||
|
#endif // BUILD_BACKEND_FTD2XX
|
||||||
|
BACKEND_DIVERGENCE_LIBUSB
|
||||||
|
// the original libusb data transfer
|
||||||
|
struct libusb_transfer *read_transfer = 0;
|
||||||
struct transfer_result read_result = { .ctx = ctx, .done = true };
|
struct transfer_result read_result = { .ctx = ctx, .done = true };
|
||||||
if (ctx->read_count) {
|
if (ctx->read_count) {
|
||||||
buffer_write_byte(ctx, 0x87); /* SEND_IMMEDIATE */
|
buffer_write_byte(ctx, 0x87); /* SEND_IMMEDIATE */
|
||||||
|
@ -953,5 +1258,10 @@ error_check:
|
||||||
if (read_transfer)
|
if (read_transfer)
|
||||||
libusb_free_transfer(read_transfer);
|
libusb_free_transfer(read_transfer);
|
||||||
|
|
||||||
|
if (retval != ERROR_OK)
|
||||||
|
mpsse_purge(ctx);
|
||||||
|
|
||||||
|
BACKEND_DIVERGENCE_END
|
||||||
|
|
||||||
return retval;
|
return retval;
|
||||||
}
|
}
|
||||||
|
|
|
@ -33,6 +33,12 @@ enum ftdi_chip_type {
|
||||||
TYPE_FT4232HA,
|
TYPE_FT4232HA,
|
||||||
};
|
};
|
||||||
|
|
||||||
|
enum mpsse_backend_type {
|
||||||
|
MPSSE_BACKEND_TYPE_UNKNOWN,
|
||||||
|
MPSSE_BACKEND_TYPE_LIBUSB,
|
||||||
|
MPSSE_BACKEND_TYPE_FTD2XX,
|
||||||
|
};
|
||||||
|
|
||||||
struct mpsse_ctx;
|
struct mpsse_ctx;
|
||||||
|
|
||||||
/* Device handling */
|
/* Device handling */
|
||||||
|
|
Loading…
Reference in New Issue