org.bidib.jbidibc.ch341a.libusb.Ch341A Maven / Gradle / Ivy
package org.bidib.jbidibc.ch341a.libusb;
import java.nio.ByteBuffer;
import java.nio.ByteOrder;
import java.nio.IntBuffer;
import org.bidib.jbidibc.messages.utils.ByteUtils;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.usb4java.BufferUtils;
import org.usb4java.ConfigDescriptor;
import org.usb4java.Device;
import org.usb4java.DeviceDescriptor;
import org.usb4java.DeviceHandle;
import org.usb4java.DeviceList;
import org.usb4java.LibUsb;
import org.usb4java.LibUsbException;
import org.usb4java.Transfer;
import org.usb4java.TransferCallback;
public class Ch341A implements AutoCloseable {
private static final Logger LOGGER = LoggerFactory.getLogger(Ch341A.class);
public static final short USB_VENDOR_ID_CH341 = 0x1a86; // Dev : (1a86) QinHeng Electronics
public static final short USB_DEVICE_ID_CH341_I2C = 0x5512; // (5512) CH341A in I2C mode
public static final short USB_DEVICE_ID_CH341_UART = 0x5523; // (5523) CH341A in UART mode
/** The interface number. */
private static final byte DEFAULT_INTERFACE = 0;
private static final int DEFAULT_CONFIGURATION = 0x01;
private static final byte BULK_WRITE_ENDPOINT = 0x02;
private static final byte BULK_READ_ENDPOINT = (byte) 0x82;
private static final int DEFAULT_TIMEOUT = 300; // 300mS for USB timeouts
private static final int IN_BUF_SZ = 0x100;
private static final int EEPROM_WRITE_BUF_SZ = 0x2b; // only for 24c64 / 24c32 ??
private static final int EEPROM_READ_BULKIN_BUF_SZ = 0x20;
private static final int EEPROM_READ_BULKOUT_BUF_SZ = 0x65;
private static final byte mCH341A_CMD_I2C_STREAM = (byte) 0xAA;
private static final byte mCH341A_CMD_I2C_STM_SET = (byte) 0x60;
private static final byte mCH341A_CMD_I2C_STM_STA = (byte) 0x74;
private static final byte mCH341A_CMD_I2C_STM_STO = (byte) 0x75;
private static final byte mCH341A_CMD_I2C_STM_OUT = (byte) 0x80;
private static final byte mCH341A_CMD_I2C_STM_END = (byte) 0x00;
/** The communication timeout in milliseconds. */
private static final int TIMEOUT = 1000;
// CH341a READ EEPROM setup packet for the 24c64
// this needs putting into a struct to allow convenient access to individual elements
// #define CH341_EEPROM_READ_SETUP_CMD "\xaa\x74\x83\xa0\x00\x00\x74\x81\xa1\xe0\x00\x00\x06\x04\x00\x00" \
// "\x00\x00\x00\x00\x40\x00\x00\x00\x11\x4d\x40\x77\xcd\xab\xba\xdc" \
// "\xaa\xe0\x00\x00\xc4\xf1\x12\x00\x11\x4d\x40\x77\xf0\xf1\x12\x00" \
// "\xd9\x8b\x41\x7e\x00\xf0\xfd\x7f\xf0\xf1\x12\x00\x5a\x88\x41\x7e" \
// "\xaa\xe0\x00\x00\x2a\x88\x41\x7e\x06\x04\x00\x00\x11\x4d\x40\x77" \
// "\xe8\xf3\x12\x00\x14\x00\x00\x00\x01\x00\x00\x00\x00\x00\x00\x00" \
// "\xaa\xdf\xc0\x75\x00"
// @formatter:off
private static final byte[] CH341_EEPROM_READ_SETUP_CMD =
{ (byte) 0xAA, 0x74, (byte) 0x83, (byte) 0xa0, 0x00, 0x00, 0x74, (byte) 0x81, (byte) 0xa1, (byte) 0xe0, 0x00, 0x00, 0x06, 0x04, 0x00, 0x00,
0x00, 0x00, 0x00, 0x00, 0x40, 0x00, 0x00, 0x00, 0x11, 0x4d, 0x40, 0x77, (byte) 0xcd, (byte) 0xab, (byte) 0xba, (byte) 0xdc,
(byte) 0xaa, (byte) 0xe0, 0x00, 0x00, (byte) 0xc4, (byte) 0xf1, 0x12, 0x00, 0x11, 0x4d, 0x40, 0x77, (byte) 0xf0, (byte) 0xf1, 0x12, 0x00,
(byte) 0xd9, (byte) 0x8b, 0x41, 0x7e, 0x00, (byte) 0xf0, (byte) 0xfd, 0x7f, (byte) 0xf0, (byte) 0xf1, 0x12, 0x00, 0x5a, (byte) 0x88, 0x41, 0x7e,
(byte) 0xaa, (byte) 0xe0, 0x00, 0x00, 0x2a, (byte) 0x88, 0x41, 0x7e, 0x06, 0x04, 0x00, 0x00, 0x11, 0x4d, 0x40, 0x77,
(byte) 0xe8, (byte) 0xf3, 0x12, 0x00, 0x14, 0x00, 0x00, 0x00, 0x01, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
(byte) 0xaa, (byte) 0xdf, (byte) 0xc0, 0x75, 0x00};
// @formatter:on
//
//// for 24c64
// #define CH341_EEPROM_READ_NEXT_CMD "\xaa\x74\x83\xa0\x00\x00\x74\x81\xa1\xe0\x00\x00\x10\x00\x00\x00" \
// "\x00\x00\x00\x00\x8c\xf1\x12\x00\x01\x00\x00\x00\x00\x00\x00\x00" \
// "\xaa\xe0\x00\x00\x4c\xf1\x12\x00\x5d\x22\xd7\x5a\xdc\xf1\x12\x00" \
// "\x8f\x04\x44\x7e\x30\x88\x41\x7e\xff\xff\xff\xff\x2a\x88\x41\x7e" \
// "\xaa\xe0\x00\x7e\x00\x00\x00\x00\x69\x0e\x3c\x00\x12\x01\x19\x00" \
// "\x0f\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x9c\x2e\x68\x00" \
// "\xaa\xdf\xc0\x75\x00"
// @formatter:off
private static final byte[] CH341_EEPROM_READ_NEXT_CMD =
{ (byte) 0xAA, 0x74, (byte) 0x83, (byte) 0xa0, 0x00, 0x00, 0x74, (byte) 0x81, (byte) 0xa1, (byte) 0xe0, 0x00, 0x00, 0x10, 0x00, 0x00, 0x00,
0x00, 0x00, 0x00, 0x00, (byte) 0x8c, (byte) 0xf1, 0x12, 0x00, 0x01, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
(byte) 0xaa, (byte) 0xe0, 0x00, 0x00, (byte) 0xc4, (byte) 0xf1, 0x12, 0x00, 0x5d, 0x22, (byte) 0xd7, 0x5a, (byte) 0xdc, (byte) 0xf1, 0x12, 0x00,
(byte) 0x8f, 0x04, 0x44, 0x7e, 0x30, (byte) 0x88, 0x41, 0x7e, (byte) 0xff, (byte) 0xff, (byte) 0xff, (byte) 0xff, 0x2a, (byte) 0x88, 0x41, 0x7e,
(byte) 0xaa, (byte) 0xe0, (byte) 0x00, (byte) 0x7e, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x69, (byte) 0x0e, (byte) 0x3c, (byte) 0x00, (byte) 0x12, (byte) 0x01, (byte) 0x19, (byte) 0x00,
(byte) 0x0f, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x00, (byte) 0x9c, (byte) 0x2e, (byte) 0x68, (byte) 0x00,
(byte) 0xaa, (byte) 0xdf, (byte) 0xc0, 0x75, 0x00};
// @formatter:on
// private Context context;
private DeviceHandle handle;
private boolean detachKernelDriver;
public void setLibUsbDebugLevel(int logLevel) {
LibUsb.setOption(null, LibUsb.OPTION_LOG_LEVEL, logLevel);
}
public void init(Integer logLevel) {
// context = new Context();
int result = LibUsb.init(null);
if (result != LibUsb.SUCCESS) {
throw new LibUsbException("Unable to initialize libusb.", result);
}
if (logLevel != null) {
LibUsb.setOption(null, LibUsb.OPTION_LOG_LEVEL, logLevel);
}
// // Read the device descriptor
// final DeviceDescriptor descriptor = new DeviceDescriptor();
// result = LibUsb.getDeviceDescriptor(device, descriptor);
// if (result < 0) {
// throw new LibUsbException("Unable to read device descriptor", result);
// }
}
// vendor Silicon Laboratories 10C4
// product CP2112 HID USB-to-SMBus Bridge EA90
public Device findDevice() {
Device device = findDevice(USB_VENDOR_ID_CH341, USB_DEVICE_ID_CH341_I2C);
LOGGER.info("Found device: {}", device);
if (device == null) {
throw new RuntimeException("No CH341A device found!");
}
return device;
}
public DeviceHandle openDevice(/* Device device */) {
int result = -1;
// DeviceHandle handle = new DeviceHandle();
//
// device = LibUsb.refDevice(device);
//
// int result = LibUsb.open(device, handle);
// if (result != LibUsb.SUCCESS) {
// throw new LibUsbException("Unable to open USB device", result);
// }
// // Open test device
final DeviceHandle handle = LibUsb.openDeviceWithVidPid(null, USB_VENDOR_ID_CH341, USB_DEVICE_ID_CH341_I2C);
if (handle == null) {
throw new RuntimeException("Unable to open USB device");
}
LOGGER.info("Open device passed, handle: {}", handle);
this.handle = handle;
Device device = LibUsb.getDevice(handle);
LOGGER.info("Get device passed, device: {}", device);
LOGGER
.info("Found [{}:{}] as device [{}] on USB bus [{}]\n", USB_VENDOR_ID_CH341, USB_DEVICE_ID_CH341_I2C,
LibUsb.getDeviceAddress(device), LibUsb.getBusNumber(device));
detachKernelDriver = LibUsb.kernelDriverActive(handle, DEFAULT_INTERFACE) > 0;
if (detachKernelDriver) {
int ret = LibUsb.detachKernelDriver(handle, DEFAULT_INTERFACE);
if (ret > 0) {
LOGGER.warn("Failed to detach kernel driver: {}", ret);
throw new LibUsbException("Unable to open USB device", ret);
}
else {
LOGGER.info("Detached kernel driver.");
}
}
else {
LOGGER.info("No kernel driver active.");
}
IntBuffer currentConfig = IntBuffer.allocate(1);
result = LibUsb.getConfiguration(handle, currentConfig);
if (result != LibUsb.SUCCESS) {
LOGGER.warn("Failed to get current device configuration: {}", result);
// return NULL;
}
LOGGER.info("Current configuration: {}", currentConfig);
if (currentConfig.get(0) != DEFAULT_CONFIGURATION) {
result = LibUsb.setConfiguration(handle, DEFAULT_CONFIGURATION);
}
if (result != LibUsb.SUCCESS) {
LOGGER.warn("Failed to set device configuration to {}: {}", DEFAULT_CONFIGURATION, result);
// return NULL;
}
// Claim the ADB interface
result = LibUsb.claimInterface(handle, DEFAULT_INTERFACE);
if (result != LibUsb.SUCCESS) {
throw new LibUsbException("Unable to claim interface", result);
}
LOGGER.info("Claim interface passed.");
ByteBuffer ch341DescriptorBuffer = ByteBuffer.allocateDirect(0x12);
result = LibUsb.getDescriptor(handle, LibUsb.DT_DEVICE, (byte) 0x00, ch341DescriptorBuffer);
if (result < LibUsb.SUCCESS) {
// throw new LibUsbException("Unable to claim interface", result);
LOGGER.warn("Unable to get the device descriptor, result: {}", result);
}
else {
LOGGER
.info(String
.format("Device reported its revision [%d.%02d]", ch341DescriptorBuffer.get(13),
ch341DescriptorBuffer.get(12)));
byte[] temp = new byte[0x12];
ch341DescriptorBuffer.get(temp);
LOGGER.info("Device descriptor: {}", ByteUtils.bytesToHex(temp));
}
try {
// Use device handle here
ConfigDescriptor descriptor = new ConfigDescriptor();
result = LibUsb.getActiveConfigDescriptor(device, descriptor);
if (result != LibUsb.SUCCESS) {
throw new LibUsbException("Unable to get the config descriptor", result);
}
LOGGER.info("Current config descriptor: {}", descriptor);
}
finally {
// LOGGER.info("Release the interface.");
// result = LibUsb.releaseInterface(handle, DEFAULT_INTERFACE);
// if (result != LibUsb.SUCCESS) {
// // throw new LibUsbException("Unable to release interface", result);
// LOGGER.warn("Unable to release interface: {}", result);
// }
//
// // Attach the kernel driver again if needed
// if (detach) {
// result = LibUsb.attachKernelDriver(handle, DEFAULT_INTERFACE);
// if (result != LibUsb.SUCCESS) {
// LOGGER.warn("Unable to re-attach kernel driver: {}", result);
// // throw new LibUsbException("Unable to re-attach kernel driver", result);
// }
// }
//
// LibUsb.close(handle);
}
return handle;
}
public void close(DeviceHandle handle) {
int result = -1;
LOGGER.info("Release the interface.");
result = LibUsb.releaseInterface(handle, DEFAULT_INTERFACE);
if (result != LibUsb.SUCCESS) {
// throw new LibUsbException("Unable to release interface", result);
LOGGER.warn("Unable to release interface: {}", result);
}
// Attach the kernel driver again if needed
if (detachKernelDriver) {
LOGGER.info("Re-attach the kernel driver.");
result = LibUsb.attachKernelDriver(handle, DEFAULT_INTERFACE);
if (result != LibUsb.SUCCESS) {
LOGGER.warn("Unable to re-attach kernel driver: {}", result);
// throw new LibUsbException("Unable to re-attach kernel driver", result);
}
// reset the flag
detachKernelDriver = false;
}
LibUsb.close(handle);
// reset the handle
handle = null;
}
/**
* @param handle
* the device handle
* @param speed
* set the i2c bus speed (speed: 0 = 20kHz; 1 = 100kHz, 2 = 400kHz, 3 = 750kHz)
*/
public void setStreamSpeed(final DeviceHandle handle, int speed) {
LOGGER.info("Set the stream speed: {}", speed);
IntBuffer transferred = IntBuffer.allocate(1);
byte[] buffer = new byte[3];
buffer[0] = mCH341A_CMD_I2C_STREAM;
buffer[1] = (byte) (mCH341A_CMD_I2C_STM_SET | (speed & 0x3));
buffer[2] = mCH341A_CMD_I2C_STM_END;
ByteBuffer data = BufferUtils.allocateByteBuffer(/* EEPROM_READ_BULKOUT_BUF_SZ */buffer.length);
data.put(buffer);
int result = LibUsb.bulkTransfer(handle, BULK_WRITE_ENDPOINT, data, transferred, DEFAULT_TIMEOUT);
if (result < LibUsb.SUCCESS) {
LOGGER.warn("Set the stream speed failed, result: {}", result);
}
LOGGER.info("Set the stream speed passed. Total transferred bytes: {}", transferred.get());
}
/**
* Flag set during the asynchronous transfers to indicate the program is finished.
*/
private volatile boolean exit = false;
private volatile int getnextpacket = 0;
private int syncackpkt;
private static final class ReadBufferHolder {
private final byte[] readBuffer;
private int offset;
public ReadBufferHolder(byte[] readBuffer) {
this.readBuffer = readBuffer;
}
public void append(byte[] buffer) {
int len = buffer.length;
System.arraycopy(buffer, 0, readBuffer, offset, len);
offset += len;
}
public byte[] getReadBuffer() {
return readBuffer;
}
public int getReceivedCount() {
return offset;
}
}
public byte[] ch341readEEPROM(final DeviceHandle handle, int bytestoread) {
byte[] ch341outBuffer = new byte[EEPROM_READ_BULKOUT_BUF_SZ];
byte[] ch341inBuffer = new byte[EEPROM_READ_BULKIN_BUF_SZ /* IN_BUF_SZ */]; // 0x100 bytes
ByteBuffer bufferIn = BufferUtils.allocateByteBuffer(ch341inBuffer.length)/* .order(ByteOrder.LITTLE_ENDIAN) */;
ByteBuffer bufferOut = BufferUtils.allocateByteBuffer(ch341outBuffer.length);
bufferOut.put(CH341_EEPROM_READ_SETUP_CMD);
LOGGER
.info("Allocated USB transfer structures, size of CH341_EEPROM_READ_SETUP_CMD: {}",
CH341_EEPROM_READ_SETUP_CMD.length);
// prepare the read buffer
byte[] readBuffer = new byte[bytestoread];
final ReadBufferHolder holder = new ReadBufferHolder(readBuffer);
Transfer transferIn = LibUsb.allocTransfer(0);
Transfer transferOut = LibUsb.allocTransfer(0);
final TransferCallback callbackIn = new TransferCallback() {
@Override
public void processTransfer(Transfer transfer) {
LOGGER.info("In, transfer status: {}", transfer.status());
switch (transfer.status()) {
case LibUsb.TRANSFER_COMPLETED:
getnextpacket = 1;
break;
default:
LOGGER.warn("In, transfer failed: {}", transfer.status());
getnextpacket = -1;
break;
}
LOGGER.info("{} bytes received", transfer.actualLength());
// prepare the buffer for the received data
byte[] ch341Received = new byte[transfer.actualLength()];
transfer.buffer().get(ch341Received);
LOGGER
.info("Data received, len: {}, data: {}", transfer.actualLength(),
ByteUtils.bytesToHex(ch341Received));
transfer.buffer().clear();
if (transfer.userData() != null) {
ReadBufferHolder holder = (ReadBufferHolder) transfer.userData();
LOGGER.info("Append the received data to the ReadBufferHolder.");
holder.append(ch341Received);
}
}
};
final TransferCallback callbackOut = new TransferCallback() {
@Override
public void processTransfer(Transfer transfer) {
LOGGER.info("Out, Sync/Ack received, status {}\n", transfer.status());
LOGGER.info("{} bytes sent.", transfer.actualLength());
syncackpkt = 1;
}
};
LibUsb.fillBulkTransfer(transferIn, handle, BULK_READ_ENDPOINT, bufferIn, callbackIn, holder, TIMEOUT);
LibUsb.fillBulkTransfer(transferOut, handle, BULK_WRITE_ENDPOINT, bufferOut, callbackOut, null, TIMEOUT);
LOGGER.info("Filled USB transfer structures");
int result = -1;
result = LibUsb.submitTransfer(transferIn);
LOGGER.info("Submitted BULK IN start packet, result: {}", result);
result = LibUsb.submitTransfer(transferOut);
LOGGER.info("Submitted BULK OUT setup packet, result: {}", result);
long start = System.currentTimeMillis();
long timeout = 100;
int readpktcount = 0;
int byteoffset = 0;
// TODO wait for result data
while (!exit) {
LOGGER.info("Read [{}] of [{}] bytes", byteoffset, bytestoread);
IntBuffer completed = BufferUtils.allocateIntBuffer();
result = LibUsb.handleEventsTimeoutCompleted(null, timeout, completed);
LOGGER.info("After handle events timeout, result: {}, completed: {}", result, completed.get());
if (result != LibUsb.SUCCESS || getnextpacket == -1) { // indicates error
LOGGER.warn("Wait for timeout failed, result: {}", result);
exit = true;
}
if (getnextpacket == 1) {
getnextpacket = 0;
readpktcount++; // increment the read packet counter
// byteoffset += EEPROM_READ_BULKIN_BUF_SZ;
byteoffset = holder.getReceivedCount();
if (byteoffset == bytestoread) {
break;
}
LOGGER.info("Re-submitting transfer request to BULK IN endpoint.");
result = LibUsb.submitTransfer(transferIn);
LOGGER.info("Submitted BULK IN start packet, result: {}", result);
if (syncackpkt > 0) {
LOGGER.info("Reset the syncackpkt.");
syncackpkt = 0;
}
// if 4th packet received, we are at end of 0x80 byte data block,
// if it is not the last block, then resubmit request for data
if (readpktcount == 4) {
LOGGER
.info(
"Submitting next transfer request to BULK OUT endpoint, CH341_EEPROM_READ_NEXT_CMD.length: {}",
CH341_EEPROM_READ_NEXT_CMD.length);
readpktcount = 0;
// TODO if the buffer is not allocated new here we get an exception
bufferOut = BufferUtils.allocateByteBuffer(ch341outBuffer.length);
bufferOut.put(CH341_EEPROM_READ_NEXT_CMD);
bufferOut.put(4, (byte) (byteoffset >> 8 & 0xff));// MSB (big-endian) byte address
bufferOut.put(5, (byte) (byteoffset & 0xff)); // LSB of 16-bit byte address
LibUsb
.fillBulkTransfer(transferOut, handle, BULK_WRITE_ENDPOINT, bufferOut, callbackOut, null,
TIMEOUT);
LibUsb.submitTransfer(transferOut); // update transfer struct (with new EEPROM page offset)
// and re-submit next transfer request to BULK OUT endpoint
}
}
}
long end = System.currentTimeMillis();
LOGGER.info("Read eeprom finished, duration: {}ms", (end - start));
// finished, free the transfer objects
LibUsb.freeTransfer(transferIn);
LibUsb.freeTransfer(transferOut);
LOGGER.info("Received data: {}", ByteUtils.bytesToHex(readBuffer));
return readBuffer;
}
/**
* Asynchronously writes some data to the device.
*
* @param handle
* The device handle.
* @param data
* The data to send to the device.
* @param callback
* The callback to execute when data has been transfered.
*/
public void write(DeviceHandle handle, byte[] data, TransferCallback callback) {
ByteBuffer buffer = BufferUtils.allocateByteBuffer(data.length);
buffer.put(data);
Transfer transfer = LibUsb.allocTransfer();
// LibUsb.fillBulkTransfer(transfer, handle, OUT_ENDPOINT, buffer, callback, null, TIMEOUT);
LOGGER.info("Sending {} bytes to device", data.length);
int result = LibUsb.submitTransfer(transfer);
if (result != LibUsb.SUCCESS) {
throw new LibUsbException("Unable to submit transfer", result);
}
}
/**
* Asynchronously reads some data from the device.
*
* @param handle
* The device handle.
* @param size
* The number of bytes to read from the device.
* @param callback
* The callback to execute when data has been received.
*/
public void read(DeviceHandle handle, int size, TransferCallback callback) {
ByteBuffer buffer = BufferUtils.allocateByteBuffer(size).order(ByteOrder.LITTLE_ENDIAN);
Transfer transfer = LibUsb.allocTransfer();
// LibUsb.fillBulkTransfer(transfer, handle, IN_ENDPOINT, buffer, callback, null, TIMEOUT);
System.out.println("Reading " + size + " bytes from device");
int result = LibUsb.submitTransfer(transfer);
if (result != LibUsb.SUCCESS) {
throw new LibUsbException("Unable to submit transfer", result);
}
}
public Device findDevice(short vendorId, short productId) {
// Read the USB device list
DeviceList list = new DeviceList();
int result = LibUsb.getDeviceList(null, list);
if (result < 0) {
throw new LibUsbException("Unable to get device list", result);
}
LOGGER.info("Found devices: {}", list);
try {
// Iterate over all devices and scan for the right one
for (Device device : list) {
DeviceDescriptor descriptor = new DeviceDescriptor();
result = LibUsb.getDeviceDescriptor(device, descriptor);
if (result != LibUsb.SUCCESS) {
throw new LibUsbException("Unable to read device descriptor", result);
}
LOGGER.info("Current device: {}", descriptor);
if (descriptor.idVendor() == vendorId && descriptor.idProduct() == productId) {
int speed = LibUsb.getDeviceSpeed(device);
LOGGER.info("Current device speed: {}", speed);
return device;
}
}
}
catch (Exception ex) {
LOGGER.warn("Failed to load the device descriptot.", ex);
}
finally {
// Ensure the allocated device list is freed
LibUsb.freeDeviceList(list, true);
}
// Device not found
return null;
}
@Override
public void close() throws Exception {
if (handle != null) {
close(handle);
}
LibUsb.exit(null);
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy