org.bidib.jbidibc.usbi2c.adapter.Cp2112UsbI2cAdapter Maven / Gradle / Ivy
/*
* Copyright (c) 2019 Victor Antonovich
*
* This work is free software; you can redistribute it and/or
* modify it under the terms of the GNU Lesser General Public License
* as published by the Free Software Foundation; either version 2.1
* of the License, or (at your option) any later version.
*
* This work is distributed in the hope that it will be useful, but
* without any warranty; without even the implied warranty of merchantability
* or fitness for a particular purpose. See the GNU Lesser General Public
* License for more details.
*
* You should have received a copy of the GNU Lesser General Public License
* along with this library; if not, write to the Free Software Foundation, Inc.,
* 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA.
*/
package org.bidib.jbidibc.usbi2c.adapter;
import java.io.IOException;
import org.bidib.jbidibc.usbhid.UsbConstants;
import org.bidib.jbidibc.usbhid.UsbDevice;
import org.bidib.jbidibc.usbhid.UsbEndpoint;
import org.bidib.jbidibc.usbhid.UsbInterface;
import org.bidib.jbidibc.usbi2c.UsbI2cManager;
import org.bidib.jbidibc.usbi2c.UsbI2cManager.UsbDeviceIdentifier;
/**
* The Silicon Labs CP2112 chip is a USB HID device which provides an SMBus controller for talking to slave devices.
*
* Data Sheet: http://www.silabs.com/Support%20Documents/TechnicalDocs/CP2112.pdf Programming Interface Specification:
* https://www.silabs.com/documents/public/application-notes/an495-cp2112-interface-specification.pdf
*/
public class Cp2112UsbI2cAdapter extends BaseUsbI2cAdapter {
// HID feature request GET_REPORT code
private static final int HID_FEATURE_REQUEST_REPORT_GET = 0x01;
// HID feature request SET_REPORT code
private static final int HID_FEATURE_REQUEST_REPORT_SET = 0x09;
// HID feature request INPUT report type
private static final int HID_FEATURE_REQUEST_REPORT_TYPE_INPUT = 0x100;
// HID feature request OUTPUT report type
private static final int HID_FEATURE_REQUEST_REPORT_TYPE_OUTPUT = 0x300;
// USB Interrupt IN/OUT packet max size
private static final int MAX_INTERRUPT_TRANSFER_SIZE = 64;
private final byte[] buffer = new byte[MAX_INTERRUPT_TRANSFER_SIZE];
// CP2112 report IDs
private static final byte REPORT_ID_SMBUS_CONFIG = 0x06;
private static final byte REPORT_ID_DATA_READ_REQUEST = 0x10;
private static final byte REPORT_ID_DATA_WRITE_READ_REQUEST = 0x11;
private static final byte REPORT_ID_DATA_READ_FORCE_SEND = 0x12;
private static final byte REPORT_ID_DATA_READ_RESPONSE = 0x13;
private static final byte REPORT_ID_DATA_WRITE_REQUEST = 0x14;
private static final byte REPORT_ID_TRANSFER_STATUS_REQUEST = 0x15;
private static final byte REPORT_ID_TRANSFER_STATUS_RESPONSE = 0x16;
private static final byte REPORT_ID_CANCEL_TRANSFER = 0x17;
// CP2112 SMBus configuration size
private static final int SMBUS_CONFIG_SIZE = 13;
// CP2112 SMBus configuration: Clock Speed
private static final int SMBUS_CONFIG_CLOCK_SPEED_OFFSET = 1;
// CP2112 SMBus configuration: Retry Time
private static final int SMBUS_CONFIG_RETRY_TIME_OFFSET = 12;
// CP2112 max I2C data write length (single write transfer)
private static final int MAX_DATA_WRITE_LENGTH = 61;
// CP2112 max I2C data read length (multiple read transfers)
private static final int MAX_DATA_READ_LENGTH = 512;
private static final int NUM_DRAIN_DATA_REPORTS_RETRIES = 10;
// CP2112 max data transfer status reads while waiting for transfer completion
private static final int NUM_TRANSFER_STATUS_RETRIES = 10;
// CP2112 transfer status codes
private static final int TRANSFER_STATUS_IDLE = 0x00;
private static final int TRANSFER_STATUS_BUSY = 0x01;
private static final int TRANSFER_STATUS_COMPLETE = 0x02;
private static final int TRANSFER_STATUS_ERROR = 0x03;
// CP2112 min clock speed
private static final int MIN_CLOCK_SPEED = 10000;
// CP2112 max clock speed
private static final int MAX_CLOCK_SPEED = CLOCK_SPEED_HIGH;
class Cp2112UsbI2cDevice extends BaseUsbI2cDevice {
Cp2112UsbI2cDevice(int address) {
super(address);
}
@Override
protected void deviceReadReg(int reg, byte[] buffer, int length) throws IOException {
readRegData(address, reg, buffer, length);
}
@Override
protected void deviceRead(byte[] buffer, int length) throws IOException {
readData(address, buffer, length);
}
@Override
protected void deviceWrite(byte[] buffer, int length) throws IOException {
writeData(address, buffer, length);
}
}
public Cp2112UsbI2cAdapter(UsbI2cManager manager, UsbDevice usbDevice) {
super(manager, usbDevice);
}
@Override
public Cp2112UsbI2cDevice getDeviceImpl(int address) {
return new Cp2112UsbI2cDevice(address);
}
@Override
protected void init(UsbDevice usbDevice) throws IOException {
if (usbDevice.getInterfaceCount() == 0) {
throw new IOException("No interfaces found for device: " + usbDevice);
}
UsbInterface usbInterface = usbDevice.getInterface(0);
if (usbInterface.getEndpointCount() < 2) {
throw new IOException("No endpoints found for device: " + usbDevice);
}
UsbEndpoint usbReadEndpoint = null, usbWriteEndpoint = null;
for (int i = 0; i < usbInterface.getEndpointCount(); i++) {
UsbEndpoint usbEndpoint = usbInterface.getEndpoint(i);
if (usbEndpoint.getType() == UsbConstants.USB_ENDPOINT_XFER_INT) {
if (usbEndpoint.getDirection() == UsbConstants.USB_DIR_IN) {
usbReadEndpoint = usbEndpoint;
}
else {
usbWriteEndpoint = usbEndpoint;
}
}
}
if (usbReadEndpoint == null || usbWriteEndpoint == null) {
throw new IOException("No read or write HID endpoint found for device: " + usbDevice);
}
setBulkEndpoints(usbReadEndpoint, usbWriteEndpoint);
configure();
// Drain all stale data reports
drainPendingDataReports();
}
@Override
protected void configure() throws IOException {
// Get current config
getHidFeatureReport(REPORT_ID_SMBUS_CONFIG, buffer, SMBUS_CONFIG_SIZE + 1); // reserve one byte for Report ID
// Clock Speed (in Hertz, default 0x000186A0 - 100 kHz)
int clockSpeed = getClockSpeed();
buffer[SMBUS_CONFIG_CLOCK_SPEED_OFFSET] = (byte) (clockSpeed >> 24);
buffer[SMBUS_CONFIG_CLOCK_SPEED_OFFSET + 1] = (byte) (clockSpeed >> 16);
buffer[SMBUS_CONFIG_CLOCK_SPEED_OFFSET + 2] = (byte) (clockSpeed >> 8);
buffer[SMBUS_CONFIG_CLOCK_SPEED_OFFSET + 3] = (byte) clockSpeed;
// Retry Time (number of retries, default 0 - no limit)
buffer[SMBUS_CONFIG_RETRY_TIME_OFFSET] = 0x00;
buffer[SMBUS_CONFIG_RETRY_TIME_OFFSET + 1] = 0x01;
sendHidFeatureReport(REPORT_ID_SMBUS_CONFIG, buffer, SMBUS_CONFIG_SIZE + 1);
}
@Override
public boolean isClockSpeedSupported(int speed) {
return (speed >= MIN_CLOCK_SPEED && speed <= MAX_CLOCK_SPEED);
}
private void checkWriteDataLength(int length) {
checkDataLength(length, MAX_DATA_WRITE_LENGTH);
}
private void checkReadDataLength(int length) {
checkDataLength(length, MAX_DATA_READ_LENGTH);
}
private void checkDataLength(int length, int maxLength) {
if (length < 1 || length > maxLength) {
throw new IllegalArgumentException(
String.format("Invalid data length: %d (min 1, max %d)", length, maxLength));
}
}
private void writeData(int address, byte[] data, int length) throws IOException {
checkWriteDataLength(length);
buffer[0] = REPORT_ID_DATA_WRITE_REQUEST;
buffer[1] = getAddressByte(address, false);
buffer[2] = (byte) length;
System.arraycopy(data, 0, buffer, 3, length);
sendHidDataReport(buffer, length + 3, USB_TIMEOUT_MILLIS);
waitTransferComplete();
}
private void readData(int address, byte[] data, int length) throws IOException {
checkReadDataLength(length);
buffer[0] = REPORT_ID_DATA_READ_REQUEST;
buffer[1] = getAddressByte(address, false); // read bit is not required
buffer[2] = (byte) ((length >> 8) & 0xff);
buffer[3] = (byte) length;
sendHidDataReport(buffer, 4, USB_TIMEOUT_MILLIS);
waitTransferComplete();
readDataFully(data, length);
}
private void readRegData(int address, int reg, byte[] data, int length) throws IOException {
checkReadDataLength(length);
buffer[0] = REPORT_ID_DATA_WRITE_READ_REQUEST;
buffer[1] = getAddressByte(address, false); // read bit is not required
buffer[2] = (byte) ((length >> 8) & 0xff);
buffer[3] = (byte) length;
buffer[4] = 0x01; // number of bytes in target address (register ID)
buffer[5] = (byte) reg;
sendHidDataReport(buffer, 6, USB_TIMEOUT_MILLIS);
waitTransferComplete();
readDataFully(data, length);
}
private void readDataFully(byte[] data, int length) throws IOException {
int totalReadLen = 0;
while (totalReadLen < length) {
sendForceReadDataRequest(length - totalReadLen);
getHidDataReport(buffer, USB_TIMEOUT_MILLIS);
if (buffer[0] != REPORT_ID_DATA_READ_RESPONSE) {
throw new IOException(String.format("Unexpected data read report ID: 0x%02x", buffer[0]));
}
if (buffer[1] == TRANSFER_STATUS_ERROR) {
throw new IOException(String.format("Data read status error, condition: 0x%02x", buffer[2]));
}
int lastReadLen = buffer[2] & 0xff;
if (lastReadLen > length - totalReadLen) {
throw new IOException(String
.format("Too many data read: " + "%d byte(s), expected: %d byte(s)", lastReadLen,
length - totalReadLen));
}
System.arraycopy(buffer, 3, data, totalReadLen, lastReadLen);
totalReadLen += lastReadLen;
}
}
private void waitTransferComplete() throws IOException {
int tryNum = 1;
while (tryNum++ <= NUM_TRANSFER_STATUS_RETRIES) {
sendTransferStatusRequest();
getHidDataReport(buffer, USB_TIMEOUT_MILLIS);
if (buffer[0] != REPORT_ID_TRANSFER_STATUS_RESPONSE) {
throw new IOException(String.format("Unexpected transfer status report ID: 0x%02x", buffer[0]));
}
switch (buffer[1]) {
case TRANSFER_STATUS_BUSY:
continue;
case TRANSFER_STATUS_COMPLETE:
return;
default:
throw new IOException(String.format("Invalid transfer status: 0x%02x", buffer[1]));
}
}
// Retries limit was reached and TRANSFER_STATUS_COMPLETE status is not reached
cancelTransfer();
throw new IOException("Transfer retries limit reached");
}
private void sendForceReadDataRequest(int length) throws IOException {
buffer[0] = REPORT_ID_DATA_READ_FORCE_SEND;
buffer[1] = (byte) ((length >> 8) & 0xff);
buffer[2] = (byte) length;
sendHidDataReport(buffer, 3, USB_TIMEOUT_MILLIS);
}
private void sendTransferStatusRequest() throws IOException {
buffer[0] = REPORT_ID_TRANSFER_STATUS_REQUEST;
buffer[1] = 0x01;
sendHidDataReport(buffer, 2, USB_TIMEOUT_MILLIS);
}
private void cancelTransfer() throws IOException {
buffer[0] = REPORT_ID_CANCEL_TRANSFER;
buffer[1] = 0x01;
// sendHidDataReport(buffer, 2, USB_TIMEOUT_MILLIS);
urbInterrupt(buffer, MAX_INTERRUPT_TRANSFER_SIZE, USB_TIMEOUT_MILLIS);
}
private void drainPendingDataReports() throws IOException {
int tryNum = 1;
while (tryNum++ <= NUM_DRAIN_DATA_REPORTS_RETRIES) {
try {
getHidDataReport(buffer, 5);
}
catch (IOException e) {
break;
}
}
if (tryNum >= NUM_DRAIN_DATA_REPORTS_RETRIES) {
throw new IOException("Can't drain pending data reports");
}
cancelTransfer();
}
private void checkReportId(int reportId) {
if ((reportId & 0xff) != reportId) {
throw new IllegalArgumentException("Invalid report ID: " + reportId);
}
}
/**
* Read HID feature report from USB device to data buffer.
*
* @param reportId
* feature report ID
* @param data
* data buffer to read report into
* @param length
* feature report data length to read
* @throws IOException
* in case of I/O error
*/
private void getHidFeatureReport(int reportId, byte[] data, int length) throws IOException {
checkReportId(reportId);
controlTransfer(
UsbConstants.USB_TYPE_CLASS | UsbConstants.USB_DIR_IN | UsbConstants.USB_INTERFACE_SUBCLASS_BOOT,
HID_FEATURE_REQUEST_REPORT_GET, reportId | HID_FEATURE_REQUEST_REPORT_TYPE_OUTPUT, 0, data, length);
}
/**
* Send HID feature report from data buffer to USB device.
*
* @param reportId
* feature report ID
* @param data
* feature report data buffer
* @param length
* feature report data length to send
* @throws IOException
* in case of I/O error
*/
private void sendHidFeatureReport(int reportId, byte[] data, int length) throws IOException {
checkReportId(reportId);
controlTransfer(
UsbConstants.USB_TYPE_CLASS | UsbConstants.USB_DIR_OUT | UsbConstants.USB_INTERFACE_SUBCLASS_BOOT,
HID_FEATURE_REQUEST_REPORT_SET, reportId | HID_FEATURE_REQUEST_REPORT_TYPE_OUTPUT, 0, data, length);
}
/**
* Read HID data report from USB device to data buffer.
*
* @param data
* data buffer to read report into
* @param timeout
* read timeout in milliseconds
* @throws IOException
* in case of data report read error or timeout
*/
private void getHidDataReport(byte[] data, int timeout) throws IOException {
bulkRead(data, 0, data.length, timeout);
}
/**
* Send HID data report from data buffer to USB device.
*
* @param data
* data buffer to send report from
* @param length
* data report length
* @param timeout
* send timeout in milliseconds
* @throws IOException
* in case of data report send error
*/
private void sendHidDataReport(byte[] data, int length, int timeout) throws IOException {
bulkWrite(data, length, USB_TIMEOUT_MILLIS);
}
public static UsbDeviceIdentifier[] getSupportedUsbDeviceIdentifiers() {
return new UsbDeviceIdentifier[] { new UsbDeviceIdentifier(0x10c4, 0xea90) };
}
}