openFPGALoader/src/anlogicCable.cpp

300 lines
6.6 KiB
C++

// SPDX-License-Identifier: Apache-2.0
/*
* Copyright (C) 2020 Gwenhael Goavec-Merou <gwenhael.goavec-merou@trabucayre.com>
*/
#include <libusb.h>
#include <stdio.h>
#include <string.h>
#include <iostream>
#include <map>
#include <vector>
#include <string>
#include "anlogicCable.hpp"
#include "display.hpp"
using namespace std;
#define ANLOGICCABLE_VID 0x0547
#define ANLOGICCABLE_PID 0x1002
#define ANLOGICCABLE_CONF_EP 0x08
#define ANLOGICCABLE_WRITE_EP 0x06
#define ANLOGICCABLE_READ_EP 0x82
#define ANLOGICCABLE_FREQ_CMD 0x01
enum analogicCablePin {
ANLOGICCABLE_TCK_PIN = (1 << 2),
ANLOGICCABLE_TDI_PIN = (1 << 1),
ANLOGICCABLE_TMS_PIN = (1 << 0)
};
enum analogicCableFreq {
ANLOGICCABLE_FREQ_6M = 0,
ANLOGICCABLE_FREQ_3M = 0x4,
ANLOGICCABLE_FREQ_2M = 0x8,
ANLOGICCABLE_FREQ_1M = 0x14,
ANLOGICCABLE_FREQ_600K = 0x24,
ANLOGICCABLE_FREQ_400K = 0x38,
ANLOGICCABLE_FREQ_200K = 0x70,
ANLOGICCABLE_FREQ_100K = 0xe8,
ANLOGICCABLE_FREQ_90K = 0xff
};
AnlogicCable::AnlogicCable(uint32_t clkHZ):
dev_handle(NULL), usb_ctx(NULL)
{
int ret;
if (libusb_init(&usb_ctx) < 0) {
cerr << "libusb init failed" << endl;
throw std::exception();
}
dev_handle = libusb_open_device_with_vid_pid(usb_ctx,
ANLOGICCABLE_VID, ANLOGICCABLE_PID);
if (!dev_handle) {
cerr << "fails to open device" << endl;
libusb_exit(usb_ctx);
throw std::exception();
}
ret = libusb_claim_interface(dev_handle, 0);
if (ret) {
cerr << "libusb error while claiming DirtyJTAG interface #0" << endl;
libusb_close(dev_handle);
libusb_exit(usb_ctx);
throw std::exception();
}
if (setClkFreq(clkHZ) < 0) {
cerr << "Fail to set frequency" << endl;
throw std::exception();
}
}
AnlogicCable::~AnlogicCable()
{
if (dev_handle)
libusb_close(dev_handle);
if (usb_ctx)
libusb_exit(usb_ctx);
}
int AnlogicCable::setClkFreq(uint32_t clkHZ)
{
int actual_length;
int ret, req_freq = clkHZ;
uint8_t buf[] = {ANLOGICCABLE_FREQ_CMD, 0};
if (clkHZ > 6000000) {
printWarn("Anlogic JTAG probe limited to 6MHz");
clkHZ = 6000000;
}
if (clkHZ >= 6000000) {
buf[1] = ANLOGICCABLE_FREQ_6M;
clkHZ = 6000000;
} else if (clkHZ >= 3000000) {
buf[1] = ANLOGICCABLE_FREQ_3M;
clkHZ = 3000000;
} else if (clkHZ >= 1000000) {
buf[1] = ANLOGICCABLE_FREQ_1M;
clkHZ = 1000000;
} else if (clkHZ >= 600000) {
buf[1] = ANLOGICCABLE_FREQ_600K;
clkHZ = 600000;
} else if (clkHZ >= 400000) {
buf[1] = ANLOGICCABLE_FREQ_400K;
clkHZ = 400000;
} else if (clkHZ >= 200000) {
buf[1] = ANLOGICCABLE_FREQ_200K;
clkHZ = 200000;
} else if (clkHZ >= 100000) {
buf[1] = ANLOGICCABLE_FREQ_100K;
clkHZ = 100000;
} else if (clkHZ >= 90000) {
buf[1] = ANLOGICCABLE_FREQ_90K;
clkHZ = 90000;
}
ret = libusb_bulk_transfer(dev_handle, ANLOGICCABLE_CONF_EP,
buf, 2, &actual_length, 1000);
if (ret < 0) {
cerr << "setClkFreq: usb bulk write failed " << ret << endl;
return -EXIT_FAILURE;
}
printWarn("Jtag frequency : requested " + std::to_string(req_freq) +
"Hz -> real " + std::to_string(clkHZ) + "Hz");
_clkHZ = clkHZ;
return clkHZ;
}
int AnlogicCable::writeTMS(uint8_t *tms, uint32_t len, bool flush_buffer)
{
(void) flush_buffer;
if (len == 0)
return 0;
uint8_t buf[512];
uint8_t mask = (ANLOGICCABLE_TCK_PIN << 4);
int full_len = len;
uint8_t *tx_ptr = tms;
while (full_len > 0) {
/* when len > buffer capacity -> limit to capacity
* else use len
*/
int xfer_len = (full_len > 512) ? 512 : full_len;
for (int i = 0; i < xfer_len; i++) {
buf[i] = mask;
if (tx_ptr[i >> 3] & (1 << (i & 0x07)))
buf[i] |= (ANLOGICCABLE_TMS_PIN |
(ANLOGICCABLE_TMS_PIN << 4));
}
/* when last burst, complite rest of buffer with
* fixed state
*/
if (xfer_len < 512) {
memset(&buf[xfer_len], buf[xfer_len-1] | ANLOGICCABLE_TCK_PIN,
512-xfer_len);
}
if (write(buf, NULL, 512, 0) < 0)
return -EXIT_FAILURE;
full_len -= xfer_len;
tx_ptr += xfer_len;
}
return len;
}
int AnlogicCable::toggleClk(uint8_t tms, uint8_t tdi, uint32_t clk_len)
{
uint8_t buf[512];
uint8_t mask = ((tms) ? ANLOGICCABLE_TMS_PIN : 0) |
((tdi) ? ANLOGICCABLE_TDI_PIN : 0);
mask |= (((mask & 0x0f) << 4) | ((ANLOGICCABLE_TCK_PIN << 4)));
uint8_t last = mask | ANLOGICCABLE_TCK_PIN;
int len = clk_len;
while (len > 0) {
/* when len > buffer capacity -> limit to capacity
* else use len
*/
int xfer_len = (len > 512) ? 512 : len;
/* since value is always the same
* fill xfer_len byte with the same value
*/
memset(buf, mask, xfer_len);
/* when last burst, complite rest of buffer with
* fixed state
*/
if (xfer_len < 512)
memset(&buf[xfer_len], last, 512-xfer_len);
if (write(buf, NULL, 512, 0) < 0)
return -EXIT_FAILURE;
len -= xfer_len;
}
return EXIT_SUCCESS;
}
int AnlogicCable::flush()
{
return 0;
}
int AnlogicCable::writeTDI(uint8_t *tx, uint8_t *rx, uint32_t len, bool end)
{
uint8_t buf[512];
uint8_t mask = (ANLOGICCABLE_TCK_PIN << 4);
int full_len = len;
uint8_t *tx_ptr = tx;
uint8_t *rx_ptr = rx;
while (full_len > 0) {
/* when len > buffer capacity -> limit to capacity
* else use len
*/
int xfer_len = (full_len > 512) ? 512 : full_len;
if (!tx) {
memset(buf, mask, xfer_len);
} else {
for (int i = 0; i < xfer_len; i++) {
buf[i] = mask;
if (tx_ptr[i >> 3] & (1 << (i & 0x07)))
buf[i] |= (ANLOGICCABLE_TDI_PIN |
(ANLOGICCABLE_TDI_PIN << 4));
}
tx_ptr += (xfer_len >> 3);
}
/* when last burst, complite rest of buffer with
* fixed state
*/
if (xfer_len < 512) {
if (end) { /* set TMS high with the last bit */
buf[xfer_len-1] |= ((ANLOGICCABLE_TMS_PIN << 4) |
(ANLOGICCABLE_TMS_PIN));
}
memset(&buf[xfer_len], buf[xfer_len-1] | ANLOGICCABLE_TCK_PIN,
512-xfer_len);
}
if (write(buf, (rx)?rx_ptr:NULL, 512, xfer_len) < 0)
return -EXIT_FAILURE;
if (rx) {
rx_ptr += (xfer_len >> 3);
}
full_len -= xfer_len;
}
return EXIT_SUCCESS;
}
int AnlogicCable::write(uint8_t *in_buf, uint8_t *out_buf, int len, int rd_len)
{
int actual_length;
int ret = libusb_bulk_transfer(dev_handle, ANLOGICCABLE_WRITE_EP,
in_buf, len, &actual_length, 1000);
if (ret < 0) {
cerr << "write: usb bulk write failed " << ret << endl;
return -EXIT_FAILURE;
}
/* all write must be followed by a read */
ret = libusb_bulk_transfer(dev_handle, ANLOGICCABLE_READ_EP,
in_buf, len, &actual_length, 1000);
if (ret < 0) {
cerr << "write: usb bulk read failed " << ret << endl;
return -EXIT_FAILURE;
}
if (out_buf) {
for (int i = 0; i < rd_len; i++) {
out_buf[i >> 3] >>= 1;
if ((in_buf[i] >> 4) & 0x01)
out_buf[i >> 3] |= 0x80;
}
}
return len;
}