mirror of
https://github.com/RPCSX/rpcsx.git
synced 2025-12-06 07:12:14 +01:00
215 lines
6.3 KiB
C++
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);
|
|
}
|