rpcsx/rpcs3/Emu/Io/usb_device.cpp
2021-03-06 18:07:08 +03:00

215 lines
6.3 KiB
C++

#include "stdafx.h"
#include "Emu/System.h"
#include "Emu/Cell/lv2/sys_usbd.h"
#include "Emu/Io/usb_device.h"
#include <libusb.h>
LOG_CHANNEL(sys_usbd);
extern void LIBUSB_CALL callback_transfer(struct libusb_transfer* transfer);
//////////////////////////////////////////////////////////////////
// ALL DEVICES ///////////////////////////////////////////////////
//////////////////////////////////////////////////////////////////
void usb_device::read_descriptors()
{
}
bool usb_device::set_configuration(u8 cfg_num)
{
current_config = cfg_num;
return true;
}
bool usb_device::set_interface(u8 int_num)
{
current_interface = int_num;
return true;
}
u64 usb_device::get_timestamp()
{
return (get_system_time() - Emu.GetPauseTime());
}
//////////////////////////////////////////////////////////////////
// PASSTHROUGH DEVICE ////////////////////////////////////////////
//////////////////////////////////////////////////////////////////
usb_device_passthrough::usb_device_passthrough(libusb_device* _device, libusb_device_descriptor& desc)
: lusb_device(_device)
{
device = UsbDescriptorNode(USB_DESCRIPTOR_DEVICE, UsbDeviceDescriptor{desc.bcdUSB, desc.bDeviceClass, desc.bDeviceSubClass, desc.bDeviceProtocol, desc.bMaxPacketSize0, desc.idVendor, desc.idProduct,
desc.bcdDevice, desc.iManufacturer, desc.iProduct, desc.iSerialNumber, desc.bNumConfigurations});
}
usb_device_passthrough::~usb_device_passthrough()
{
if (lusb_handle)
libusb_close(lusb_handle);
if (lusb_device)
libusb_unref_device(lusb_device);
}
bool usb_device_passthrough::open_device()
{
if (libusb_open(lusb_device, &lusb_handle) == LIBUSB_SUCCESS)
{
#ifdef __linux__
libusb_set_auto_detach_kernel_driver(lusb_handle, true);
#endif
return true;
}
return false;
}
void usb_device_passthrough::read_descriptors()
{
// Directly getting configuration descriptors from the device instead of going through libusb parsing functions as they're not needed
for (u8 index = 0; index < device._device.bNumConfigurations; index++)
{
u8 buf[1000];
int ssize = libusb_control_transfer(lusb_handle, +LIBUSB_ENDPOINT_IN | +LIBUSB_REQUEST_TYPE_STANDARD | +LIBUSB_RECIPIENT_DEVICE, LIBUSB_REQUEST_GET_DESCRIPTOR, 0x0200 | index, 0, buf, 1000, 0);
if (ssize < 0)
{
sys_usbd.fatal("Couldn't get the config from the device: %d", ssize);
continue;
}
// Minimalistic parse
auto& conf = device.add_node(UsbDescriptorNode(buf[0], buf[1], &buf[2]));
for (int index = buf[0]; index < ssize;)
{
conf.add_node(UsbDescriptorNode(buf[index], buf[index + 1], &buf[index + 2]));
index += buf[index];
}
}
}
bool usb_device_passthrough::set_configuration(u8 cfg_num)
{
usb_device::set_configuration(cfg_num);
return (libusb_set_configuration(lusb_handle, cfg_num) == LIBUSB_SUCCESS);
};
bool usb_device_passthrough::set_interface(u8 int_num)
{
usb_device::set_interface(int_num);
return (libusb_claim_interface(lusb_handle, int_num) == LIBUSB_SUCCESS);
}
void usb_device_passthrough::control_transfer(u8 bmRequestType, u8 bRequest, u16 wValue, u16 wIndex, u16 /*wLength*/, u32 buf_size, u8* buf, UsbTransfer* transfer)
{
if (transfer->setup_buf.size() < buf_size + 8)
transfer->setup_buf.resize(buf_size + 8);
libusb_fill_control_setup(transfer->setup_buf.data(), bmRequestType, bRequest, wValue, wIndex, buf_size);
memcpy(transfer->setup_buf.data() + 8, buf, buf_size);
libusb_fill_control_transfer(transfer->transfer, lusb_handle, transfer->setup_buf.data(), callback_transfer, transfer, 0);
libusb_submit_transfer(transfer->transfer);
}
void usb_device_passthrough::interrupt_transfer(u32 buf_size, u8* buf, u32 endpoint, UsbTransfer* transfer)
{
libusb_fill_interrupt_transfer(transfer->transfer, lusb_handle, endpoint, buf, buf_size, callback_transfer, transfer, 0);
libusb_submit_transfer(transfer->transfer);
}
void usb_device_passthrough::isochronous_transfer(UsbTransfer* transfer)
{
// TODO actual endpoint
// TODO actual size?
libusb_fill_iso_transfer(transfer->transfer, lusb_handle, 0x81, static_cast<u8*>(transfer->iso_request.buf.get_ptr()), 0xFFFF, transfer->iso_request.num_packets, callback_transfer, transfer, 0);
for (u32 index = 0; index < transfer->iso_request.num_packets; index++)
{
transfer->transfer->iso_packet_desc[index].length = transfer->iso_request.packets[index];
}
libusb_submit_transfer(transfer->transfer);
}
//////////////////////////////////////////////////////////////////
// EMULATED DEVICE ///////////////////////////////////////////////
//////////////////////////////////////////////////////////////////
usb_device_emulated::usb_device_emulated()
{
}
usb_device_emulated::usb_device_emulated(const UsbDeviceDescriptor& _device)
{
device = UsbDescriptorNode(USB_DESCRIPTOR_DEVICE, _device);
}
bool usb_device_emulated::open_device()
{
return true;
}
s32 usb_device_emulated::get_descriptor(u8 type, u8 index, u8* ptr, u32 /*max_size*/)
{
if (type == USB_DESCRIPTOR_STRING)
{
if (index < strings.size())
{
u8 string_len = ::narrow<u8>(strings[index].size());
ptr[0] = (string_len * 2) + 2;
ptr[1] = USB_DESCRIPTOR_STRING;
for (u32 i = 0; i < string_len; i++)
{
ptr[2 + (i * 2)] = strings[index].data()[i];
ptr[3 + (i * 2)] = 0;
}
return ptr[0];
}
}
else
{
sys_usbd.error("[Emulated]: Trying to get a descriptor other than string descriptor");
}
return -1;
}
void usb_device_emulated::control_transfer(u8 bmRequestType, u8 bRequest, u16 wValue, u16 /*wIndex*/, u16 /*wLength*/, u32 buf_size, u8* /*buf*/, UsbTransfer* transfer)
{
transfer->fake = true;
transfer->expected_count = buf_size;
transfer->expected_result = HC_CC_NOERR;
transfer->expected_time = usb_device::get_timestamp() + 100;
switch (bmRequestType)
{
case 0:
switch (bRequest)
{
case 0x09: usb_device::set_configuration(::narrow<u8>(wValue)); break;
default: sys_usbd.fatal("Unhandled control transfer(0): 0x%x", bRequest); break;
}
break;
default: sys_usbd.fatal("Unhandled control transfer: 0x%x", bmRequestType); break;
}
}
// Temporarily
#ifndef _MSC_VER
#pragma GCC diagnostic ignored "-Wunused-parameter"
#endif
void usb_device_emulated::interrupt_transfer(u32 buf_size, u8* buf, u32 endpoint, UsbTransfer* transfer)
{
}
void usb_device_emulated::isochronous_transfer(UsbTransfer* transfer)
{
}
void usb_device_emulated::add_string(char* str)
{
strings.emplace_back(str);
}