org.bidib.jbidibc.usbi2c.adapter.Ch341UsbI2cAdapter Maven / Gradle / Ivy
/*
* Copyright (c) 2020 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;
/**
* CH341 is a USB bus convert chip, providing UART, printer port, parallel and synchronous serial with 2-wire or 4-wire
* through USB bus (http://www.anok.ceti.pl/download/ch341ds1.pdf).
*/
public class Ch341UsbI2cAdapter extends BaseUsbI2cAdapter {
private static final int CH341_I2C_LOW_SPEED = 0; // low speed - 20kHz
private static final int CH341_I2C_STANDARD_SPEED = 1; // standard speed - 100kHz
private static final int CH341_I2C_FAST_SPEED = 2; // fast speed - 400kHz
private static final int CH341_I2C_HIGH_SPEED = 3; // high speed - 750kHz
private static final int CH341_CMD_I2C_STREAM = 0xAA;
private static final int CH341_CMD_I2C_STM_STA = 0x74;
private static final int CH341_CMD_I2C_STM_STO = 0x75;
private static final int CH341_CMD_I2C_STM_OUT = 0x80;
private static final int CH341_CMD_I2C_STM_IN = 0xC0;
private static final int CH341_CMD_I2C_STM_SET = 0x60;
private static final int CH341_CMD_I2C_STM_END = 0x00;
// CH341 max transfer size
private static final int MAX_TRANSFER_SIZE = 32;
private final byte[] writeBuffer = new byte[MAX_TRANSFER_SIZE];
private final byte[] readBuffer = new byte[MAX_TRANSFER_SIZE];
class Ch341UsbI2cDevice extends BaseUsbI2cDevice {
Ch341UsbI2cDevice(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 Ch341UsbI2cAdapter(UsbI2cManager i2cManager, UsbDevice usbDevice) {
super(i2cManager, usbDevice);
}
@Override
protected Ch341UsbI2cDevice getDeviceImpl(int address) {
return new Ch341UsbI2cDevice(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_BULK) {
if (usbEndpoint.getDirection() == UsbConstants.USB_DIR_IN) {
usbReadEndpoint = usbEndpoint;
}
else {
usbWriteEndpoint = usbEndpoint;
}
}
}
if (usbReadEndpoint == null || usbWriteEndpoint == null) {
throw new IOException("No read or write bulk endpoint found for device: " + usbDevice);
}
setBulkEndpoints(usbReadEndpoint, usbWriteEndpoint);
configure();
}
protected int getClockSpeedConstant(int speed) {
switch (speed) {
case 20000:
return CH341_I2C_LOW_SPEED;
case CLOCK_SPEED_STANDARD:
return CH341_I2C_STANDARD_SPEED;
case CLOCK_SPEED_FAST:
return CH341_I2C_FAST_SPEED;
case 750000:
return CH341_I2C_HIGH_SPEED;
}
return -1;
}
@Override
public boolean isClockSpeedSupported(int speed) {
return (getClockSpeedConstant(speed) >= 0);
}
@Override
protected void configure() throws IOException {
writeBuffer[0] = (byte) CH341_CMD_I2C_STREAM;
writeBuffer[1] = (byte) (CH341_CMD_I2C_STM_SET | getClockSpeedConstant(getClockSpeed()));
writeBuffer[2] = CH341_CMD_I2C_STM_END;
writeBulkData(writeBuffer, 3);
}
private void checkDataLength(int length) {
int maxLength = MAX_TRANSFER_SIZE - 6;
if (length > maxLength) {
throw new IllegalArgumentException(String.format("Invalid data length: %d (max %d)", length, maxLength));
}
}
private void writeData(int address, byte[] data, int length) throws IOException {
checkDataLength(length);
int i = 0;
writeBuffer[i++] = (byte) CH341_CMD_I2C_STREAM;
writeBuffer[i++] = CH341_CMD_I2C_STM_STA; // START condition
writeBuffer[i++] = (byte) (CH341_CMD_I2C_STM_OUT | (length + 1)); // data length + 1
writeBuffer[i++] = getAddressByte(address, false);
System.arraycopy(data, 0, writeBuffer, i, length);
i += length;
writeBuffer[i++] = CH341_CMD_I2C_STM_STO; // STOP condition
writeBuffer[i++] = CH341_CMD_I2C_STM_END; // end of transaction
writeBulkData(writeBuffer, i);
}
private void readData(int address, byte[] data, int length) throws IOException {
checkDataLength(length);
checkDevicePresence(address); // to avoid weird phantom devices in scan results
int i = 0;
writeBuffer[i++] = (byte) CH341_CMD_I2C_STREAM;
writeBuffer[i++] = CH341_CMD_I2C_STM_STA; // START condition
writeBuffer[i++] = (byte) (CH341_CMD_I2C_STM_OUT | 1); // zero data length + 1
writeBuffer[i++] = getAddressByte(address, true);
if (length > 0) {
for (int j = 0; j < length - 1; j++) {
writeBuffer[i++] = (byte) (CH341_CMD_I2C_STM_IN | 1);
}
writeBuffer[i++] = (byte) CH341_CMD_I2C_STM_IN;
}
writeBuffer[i++] = CH341_CMD_I2C_STM_STO; // STOP condition
writeBuffer[i++] = CH341_CMD_I2C_STM_END; // end of transaction
int res = transferBulkData(writeBuffer, i, data, length);
}
private void readRegData(int address, int reg, byte[] data, int length) throws IOException {
checkDataLength(length);
// Write register number
int i = 0;
writeBuffer[i++] = (byte) CH341_CMD_I2C_STREAM;
writeBuffer[i++] = CH341_CMD_I2C_STM_STA; // START condition
writeBuffer[i++] = (byte) (CH341_CMD_I2C_STM_OUT | 2); // reg data length + 1
writeBuffer[i++] = getAddressByte(address, false);
writeBuffer[i++] = (byte) reg;
writeBulkData(writeBuffer, i);
// Read register data
i = 0;
writeBuffer[i++] = (byte) CH341_CMD_I2C_STREAM;
writeBuffer[i++] = CH341_CMD_I2C_STM_STA; // START condition
writeBuffer[i++] = (byte) (CH341_CMD_I2C_STM_OUT | 1); // zero data length + 1
writeBuffer[i++] = getAddressByte(address, true);
if (length > 0) {
for (int j = 0; j < length - 1; j++) {
writeBuffer[i++] = (byte) (CH341_CMD_I2C_STM_IN | 1);
}
writeBuffer[i++] = (byte) CH341_CMD_I2C_STM_IN;
}
writeBuffer[i++] = CH341_CMD_I2C_STM_STO; // STOP condition
writeBuffer[i++] = CH341_CMD_I2C_STM_END; // end of transaction
transferBulkData(writeBuffer, i, data, length);
}
private void checkDevicePresence(int address) throws IOException {
int i = 0;
writeBuffer[i++] = (byte) CH341_CMD_I2C_STREAM;
writeBuffer[i++] = CH341_CMD_I2C_STM_STA; // START condition
writeBuffer[i++] = (byte) CH341_CMD_I2C_STM_OUT;
writeBuffer[i++] = getAddressByte(address, true);
writeBuffer[i++] = CH341_CMD_I2C_STM_STO; // STOP condition
writeBuffer[i++] = CH341_CMD_I2C_STM_END; // end of transaction
int res = transferBulkData(writeBuffer, i, writeBuffer, 1);
if (res <= 0 || (writeBuffer[0] & 0x80) != 0) {
throw new IOException(String.format("No device present at address 0x%02x", address));
}
}
/**
* Write request and read response, if needed.
*
* @param writeData
* data buffer to write data
* @param writeLength
* data length to write
* @param readData
* data buffer to read data
* @param readLength
* data length to read (can be zero)
* @return actual length of read data
* @throws IOException
* in case of data read/write error or timeout
*/
private int transferBulkData(byte[] writeData, int writeLength, byte[] readData, int readLength)
throws IOException {
writeBulkData(writeData, writeLength);
if (readLength > 0) {
readLength = readBulkData(readBuffer, MAX_TRANSFER_SIZE);
System.arraycopy(readBuffer, 0, readData, 0, readLength);
}
return readLength;
}
/**
* Read bulk data from USB device to data buffer.
*
* @param data
* data buffer to read data
* @param length
* data length to read
* @return actual length of read data
* @throws IOException
* in case of data read error or timeout
*/
private int readBulkData(byte[] data, int length) throws IOException {
return bulkRead(data, 0, length, USB_TIMEOUT_MILLIS);
}
/**
* Write bulk data from data buffer to USB device.
*
* @param data
* data buffer to write data
* @param length
* data length to write
* @throws IOException
* in case of data write error
*/
private void writeBulkData(byte[] data, int length) throws IOException {
bulkWrite(data, length, USB_TIMEOUT_MILLIS);
}
public static UsbDeviceIdentifier[] getSupportedUsbDeviceIdentifiers() {
return new UsbDeviceIdentifier[] { new UsbDeviceIdentifier(0x1a86, 0x5512) };
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy