Bochs/bochs/iodev/usb/usb_hub.cc
Benjamin David Lunt bcbe5da030
Usb compliance (#91)
Made all devices pass USB 2.0 Compliance on WinXP (old version of the
compliance test).
USB Hub still needs a little work to be in complete compliance.
Fixed potential bug in EHCI to UHCI hand-off.
Fixed compilation error with Floppy CB/CBI emulation.
Minor syntax fixes (tabulation, old irrelevant comments)
MSD serial numbers must be 12 chars.
Added to CHANGES file
2023-10-15 07:23:18 +03:00

982 lines
31 KiB
C++

/////////////////////////////////////////////////////////////////////////
// $Id$
/////////////////////////////////////////////////////////////////////////
//
// USB hub emulation support (ported from QEMU)
//
// Copyright (C) 2005 Fabrice Bellard
// Copyright (C) 2009-2023 The Bochs Project
//
// Permission is hereby granted, free of charge, to any person obtaining a copy
// of this software and associated documentation files (the "Software"), to deal
// in the Software without restriction, including without limitation the rights
// to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
// copies of the Software, and to permit persons to whom the Software is
// furnished to do so, subject to the following conditions:
//
// The above copyright notice and this permission notice shall be included in
// all copies or substantial portions of the Software.
//
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
// IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
// FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
// THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
// LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
// OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
// THE SOFTWARE.
/////////////////////////////////////////////////////////////////////////
// Define BX_PLUGGABLE in files that can be compiled into plugins. For
// platforms that require a special tag on exported symbols, BX_PLUGGABLE
// is used to know when we are exporting symbols and when we are importing.
#define BX_PLUGGABLE
#include "iodev.h"
#if BX_SUPPORT_PCI && BX_SUPPORT_PCIUSB
#include "usb_common.h"
#include "usb_hub.h"
#define LOG_THIS
// USB device plugin entry point
PLUGIN_ENTRY_FOR_MODULE(usb_hub)
{
if (mode == PLUGIN_PROBE) {
return (int)PLUGTYPE_USB;
}
return 0; // Success
}
//
// Define the static class that registers the derived USB device class,
// and allocates one on request.
//
class bx_usb_hub_locator_c : public usbdev_locator_c {
public:
bx_usb_hub_locator_c(void) : usbdev_locator_c("usb_hub") {}
protected:
usb_device_c *allocate(const char *devname) {
return (new usb_hub_device_c());
}
} bx_usb_hub_match;
#define PORT_STAT_CONNECTION 0x0001
#define PORT_STAT_ENABLE 0x0002
#define PORT_STAT_SUSPEND 0x0004
#define PORT_STAT_OVERCURRENT 0x0008
#define PORT_STAT_RESET 0x0010
#define PORT_STAT_POWER 0x0100
#define PORT_STAT_LOW_SPEED 0x0200
#define PORT_STAT_HIGH_SPEED 0x0400
#define PORT_STAT_TEST 0x0800
#define PORT_STAT_INDICATOR 0x1000
#define PORT_STAT_C_CONNECTION 0x0001
#define PORT_STAT_C_ENABLE 0x0002
#define PORT_STAT_C_SUSPEND 0x0004
#define PORT_STAT_C_OVERCURRENT 0x0008
#define PORT_STAT_C_RESET 0x0010
#define PORT_CONNECTION 0
#define PORT_ENABLE 1
#define PORT_SUSPEND 2
#define PORT_OVERCURRENT 3
#define PORT_RESET 4
#define PORT_POWER 8
#define PORT_LOWSPEED 9
#define PORT_HIGHSPEED 10
#define PORT_C_CONNECTION 16
#define PORT_C_ENABLE 17
#define PORT_C_SUSPEND 18
#define PORT_C_OVERCURRENT 19
#define PORT_C_RESET 20
#define PORT_TEST 21
#define PORT_INDICATOR 22
static Bit32u serial_number = 1234;
// Set this to a version of the USB you wish to emulate.
// 0x0100 = v1.0 or 0x0101 = v1.1
#define USB_HUB_VERSION 0x0100
// Set USB_HUB_POWER_SWITCHING to 1 if power switching is supported
// (If not supported power is always on)
// (It must be set if USB_HUB_VERSION == 1.1+)
#define USB_HUB_POWER_SWITCHING 1
#if (USB_HUB_VERSION > 0x100) && (USB_HUB_POWER_SWITCHING == 0)
#error USB_HUB_POWER_SWITCHING must be set for USB_HUB_VERSION 1.1+
#endif
// Set USB_HUB_POWER_PER_PORT to 1 if per-port power switching is on,
// else set to 0 for ganged power switching (all ports powered at same time)
// (is ignored if USB_HUB_POWER_SWITCHING == 0)
#define USB_HUB_POWER_PER_PORT 1
// The USB specification (usb1.0 11.8, usb1.1 11.13.1, usb2.0 11.12.1)
// states that if there is nothing to report from the IN interrupt
// that the device should respond with a USB_RET_NAK. However, Win95
// considers that an error and stops the IN pipe, finding no further
// connection status. Set this to 1 if you are emulating Win95 or
// a similar Guest.
#define USB_HUB_ALWAYS_REPORT 1
// If you change any of the Max Packet Size, or other fields within these
// descriptors, you must also change the d.endpoint_info[] array
// to match your changes.
static const Bit8u bx_hub_dev_descriptor[] = {
0x12, /* u8 bLength; */
0x01, /* u8 bDescriptorType; Device */
#if USB_HUB_VERSION == 0x0100
0x00, 0x01, /* u16 bcdUSB; v1.0 */
#elif USB_HUB_VERSION == 0x0101
0x01, 0x01, /* u16 bcdUSB; v1.1 */
#else
#error Unknown USB_HUB_VERSION value found
#endif
0x09, /* u8 bDeviceClass; HUB_CLASSCODE */
0x00, /* u8 bDeviceSubClass; */
0x00, /* u8 bDeviceProtocol; */
0x40, /* u8 bMaxPacketSize; 64 Bytes */
0x09, 0x04, /* u16 idVendor; */
0x5A, 0x00, /* u16 idProduct; */
0x00, 0x01, /* u16 bcdDevice */
0x01, /* u8 iManufacturer; */
0x02, /* u8 iProduct; */
0x03, /* u8 iSerialNumber; */
0x01 /* u8 bNumConfigurations; */
};
/* XXX: patch interrupt size */
#define BX_Hub_Config_Descriptor_pos 22
static Bit8u bx_hub_config_descriptor[] = {
/* one configuration */
0x09, /* u8 bLength; */
0x02, /* u8 bDescriptorType; Configuration */
0x19, 0x00, /* u16 wTotalLength; */
0x01, /* u8 bNumInterfaces; (1) */
0x01, /* u8 bConfigurationValue; */
0x00, /* u8 iConfiguration; */
0xE0, /* u8 bmAttributes;
Bit 7: must be set,
6: Self-powered,
5: Remote wakeup,
4..0: resvd */
0x32, /* u8 MaxPower; */
/* USB 1.1:
* USB 2.0, single TT organization (mandatory):
* one interface, protocol 0
*
* USB 2.0, multiple TT organization (optional):
* two interfaces, protocols 1 (like single TT)
* and 2 (multiple TT mode) ... config is
* sometimes settable
* NOT IMPLEMENTED
*/
/* one interface */
0x09, /* u8 if_bLength; */
0x04, /* u8 if_bDescriptorType; Interface */
0x00, /* u8 if_bInterfaceNumber; */
0x00, /* u8 if_bAlternateSetting; */
0x01, /* u8 if_bNumEndpoints; */
0x09, /* u8 if_bInterfaceClass; HUB_CLASSCODE */
0x00, /* u8 if_bInterfaceSubClass; */
0x00, /* u8 if_bInterfaceProtocol; [usb1.1 or single tt] */
0x00, /* u8 if_iInterface; */
/* one endpoint (status change endpoint) */
0x07, /* u8 ep_bLength; */
0x05, /* u8 ep_bDescriptorType; Endpoint */
0x81, /* u8 ep_bEndpointAddress; IN Endpoint 1 */
0x03, /* u8 ep_bmAttributes; Interrupt */
0x02, 0x00, /* u16 ep_wMaxPacketSize; 1 + (USB_HUB_MAX_PORTS / 8) */
0xff /* u8 ep_bInterval; (255ms -- usb 2.0 spec) */
};
static const Bit8u bx_hub_hub_descriptor[] =
{
0x00, /* u8 bLength; patched in later */
0x29, /* u8 bDescriptorType; Hub-descriptor */
0x00, /* u8 bNbrPorts; (patched later) */
#if USB_HUB_POWER_SWITCHING
#if USB_HUB_POWER_PER_PORT
0x09, 0x00, /* u16 wHubCharacteristics; (per-port OC, per-port power switching) */
#else
0x08, 0x00, /* u16 wHubCharacteristics; (per-port OC, ganged power switching) */
#endif
#else
0x0a, 0x00, /* u16 wHubCharacteristics; (per-port OC, no power switching) */
#endif
0x01, /* u8 bPwrOn2pwrGood; 2ms */
0x40 /* u8 bHubContrCurrent; 64 mA */
/* DeviceRemovable and PortPwrCtrlMask patched in later */
};
static Bit8u hub_count = 0;
void usb_hub_restore_handler(void *dev, bx_list_c *conf);
usb_hub_device_c::usb_hub_device_c()
{
char pname[10];
char label[32];
d.speed = d.minspeed = d.maxspeed = USB_SPEED_FULL;
strcpy(d.devname, "Bochs USB HUB");
d.dev_descriptor = bx_hub_dev_descriptor;
d.device_desc_size = sizeof(bx_hub_dev_descriptor);
d.config_descriptor = bx_hub_config_descriptor;
d.config_desc_size = sizeof(bx_hub_config_descriptor);
d.endpoint_info[USB_CONTROL_EP].max_packet_size = 64; // Control ep0
d.endpoint_info[USB_CONTROL_EP].max_burst_size = 0;
d.endpoint_info[1].max_packet_size = (USB_HUB_MAX_PORTS + 1 + 7) / 8; // In ep1
d.endpoint_info[1].max_burst_size = 0;
d.vendor_desc = "BOCHS";
d.product_desc = "BOCHS USB HUB";
memset((void*)&hub, 0, sizeof(hub));
sprintf(hub.serial_number, "%d", serial_number++);
d.serial_num = hub.serial_number;
hub.device_change = 0;
hub.n_ports = USB_HUB_DEF_PORTS;
// prepare runtime config options
bx_list_c *usb_rt = (bx_list_c*)SIM->get_param(BXPN_MENU_RUNTIME_USB);
sprintf(pname, "exthub%u", ++hub_count);
sprintf(label, "External Hub #%u Configuration", hub_count);
hub.config = new bx_list_c(usb_rt, pname, label);
hub.config->set_options(bx_list_c::SHOW_PARENT);
hub.config->set_device_param(this);
put("usb_hub", "USBHUB");
}
usb_hub_device_c::~usb_hub_device_c(void)
{
for (int i = 0; i < hub.n_ports; i++) {
remove_device(i);
}
if (SIM->is_wx_selected()) {
bx_list_c *usb = (bx_list_c *) SIM->get_param("ports.usb");
usb->remove(hub.config->get_name());
}
bx_list_c *usb_rt = (bx_list_c *) SIM->get_param(BXPN_MENU_RUNTIME_USB);
usb_rt->remove(hub.config->get_name());
}
bool usb_hub_device_c::set_option(const char *option)
{
if (!strncmp(option, "ports:", 6)) {
hub.n_ports = atoi(&option[6]);
if ((hub.n_ports < 2) || (hub.n_ports > USB_HUB_MAX_PORTS)) {
BX_ERROR(("ignoring invalid number of ports (%d)", hub.n_ports));
hub.n_ports = USB_HUB_DEF_PORTS;
}
return 1;
}
return 0;
}
bool usb_hub_device_c::init()
{
int i;
char pname[10];
char label[32];
bx_list_c *port, *deplist;
bx_param_enum_c *device;
bx_param_string_c *options;
bx_param_bool_c *overcurrent;
// set up config descriptor, status and runtime config for hub.n_ports
bx_hub_config_descriptor[BX_Hub_Config_Descriptor_pos] = (hub.n_ports + 1 + 7) / 8;
for(i = 0; i < hub.n_ports; i++) {
hub.usb_port[i].PortStatus = PORT_STAT_POWER;
hub.usb_port[i].PortChange = 0;
}
hub.PortStatusC = 0;
for(i = 0; i < hub.n_ports; i++) {
sprintf(pname, "port%d", i+1);
sprintf(label, "Port #%d Configuration", i+1);
port = new bx_list_c(hub.config, pname, label);
port->set_options(port->SERIES_ASK | port->USE_BOX_TITLE);
device = new bx_param_enum_c(port, "device", "Device", "",
bx_usbdev_ctl.get_device_names(), 0, 0);
device->set_handler(hub_param_handler);
options = new bx_param_string_c(port, "options", "Options", "", "",
BX_PATHNAME_LEN);
options->set_enable_handler(hub_param_enable_handler);
overcurrent = new bx_param_bool_c(port,
"over_current",
"signal over-current",
"signal over-current", 0);
overcurrent->set_handler(hub_param_oc_handler);
deplist = new bx_list_c(NULL);
deplist->add(options);
deplist->add(overcurrent);
device->set_dependent_list(deplist, 1);
device->set_dependent_bitmap(0, 0);
}
if (SIM->is_wx_selected()) {
bx_list_c *usb = (bx_list_c *) SIM->get_param("ports.usb");
usb->add(hub.config);
}
sprintf(hub.info_txt, "%d-port USB hub", hub.n_ports);
d.connected = 1;
d.alt_iface_max = 0;
return 1;
}
const char *usb_hub_device_c::get_info()
{
return hub.info_txt;
}
void usb_hub_device_c::register_state_specific(bx_list_c *parent)
{
Bit8u i;
char portnum[16];
bx_list_c *port, *pconf, *config;
hub.state = new bx_list_c(parent, "hub", "USB HUB Device State");
for (i=0; i<hub.n_ports; i++) {
sprintf(portnum, "port%d", i+1);
port = new bx_list_c(hub.state, portnum);
pconf = (bx_list_c*)hub.config->get_by_name(portnum);
config = new bx_list_c(port, portnum);
config->add(pconf->get_by_name("device"));
config->add(pconf->get_by_name("options"));
config->set_restore_handler(this, usb_hub_restore_handler);
BXRS_HEX_PARAM_FIELD(port, PortStatus, hub.usb_port[i].PortStatus);
BXRS_HEX_PARAM_FIELD(port, PortChange, hub.usb_port[i].PortChange);
// empty list for USB device state
new bx_list_c(port, "device");
}
BXRS_HEX_PARAM_FIELD(hub.state, PortStatusC, hub.PortStatusC);
}
void usb_hub_device_c::after_restore_state()
{
for (int i=0; i<hub.n_ports; i++) {
if (hub.usb_port[i].device != NULL) {
hub.usb_port[i].device->after_restore_state();
}
}
}
void usb_hub_device_c::handle_reset()
{
int i;
BX_DEBUG(("Reset"));
for (i = 0; i < hub.n_ports; i++) {
hub.usb_port[i].PortStatus = PORT_STAT_POWER;
hub.usb_port[i].PortChange = 0;
if (hub.usb_port[i].device != NULL) {
hub.usb_port[i].PortStatus |= PORT_STAT_CONNECTION;
hub.usb_port[i].PortChange |= PORT_STAT_C_CONNECTION;
if (hub.usb_port[i].device->get_speed() == USB_SPEED_LOW) {
hub.usb_port[i].PortStatus |= PORT_STAT_LOW_SPEED;
}
}
}
}
int usb_hub_device_c::handle_control(int request, int value, int index, int length, Bit8u *data)
{
int ret = 0;
unsigned int n;
ret = handle_control_common(request, value, index, length, data);
if (ret >= 0) {
return ret;
}
ret = 0;
switch(request) {
case DeviceOutRequest | USB_REQ_CLEAR_FEATURE:
goto fail;
break;
case DeviceOutRequest | USB_REQ_SET_FEATURE:
goto fail;
break;
case EndpointRequest | USB_REQ_GET_STATUS:
BX_DEBUG(("USB_REQ_GET_STATUS: Endpoint."));
// if the endpoint is currently halted, return bit 0 = 1
if (value == USB_ENDPOINT_HALT) {
if (index == 0x81) {
data[0] = 0x00 | (get_halted(index) ? 1 : 0);
data[1] = 0x00;
ret = 2;
} else {
BX_ERROR(("EndpointRequest | USB_REQ_GET_STATUS: index > ep count: %d", index));
goto fail;
}
} else {
BX_ERROR(("EndpointRequest | USB_REQ_SET_FEATURE: Unknown Get Status Request found: %d", value));
goto fail;
}
break;
case DeviceRequest | USB_REQ_GET_DESCRIPTOR:
switch(value >> 8) {
case USB_DT_STRING:
switch (value & 0xFF) {
case 0xEE:
// Microsoft OS Descriptor check
// We don't support this check, so fail
BX_INFO(("USB HUB handle_control: Microsoft OS specific 0xEE string descriptor"));
goto fail;
break;
default:
BX_ERROR(("unknown string descriptor type %d", value & 0xff));
goto fail;
break;
}
break;
default:
BX_ERROR(("unknown descriptor type: 0x%02x", (value >> 8)));
goto fail;
}
break;
/* hub specific requests */
case DeviceClassInRequest | USB_REQ_GET_STATUS:
if (d.state == USB_STATE_CONFIGURED) {
data[0] = 0;
data[1] = 0;
data[2] = 0;
data[3] = 0;
ret = 4;
} else
goto fail;
break;
case OtherInClassRequest | USB_REQ_GET_STATUS:
n = index - 1;
if (n >= hub.n_ports)
goto fail;
data[0] = (hub.usb_port[n].PortStatus & 0xff);
data[1] = (hub.usb_port[n].PortStatus >> 8);
data[2] = (hub.usb_port[n].PortChange & 0xff);
data[3] = (hub.usb_port[n].PortChange >> 8);
ret = 4;
break;
case DeviceOutClassRequest | USB_REQ_SET_FEATURE:
case DeviceOutClassRequest | USB_REQ_CLEAR_FEATURE:
if (value == 0 || value == 1) {
;
} else {
goto fail;
}
ret = 0;
break;
case OtherOutClassRequest | USB_REQ_SET_FEATURE:
n = index - 1;
if (n >= hub.n_ports)
goto fail;
switch(value) {
case PORT_SUSPEND:
hub.usb_port[n].PortStatus |= PORT_STAT_SUSPEND;
break;
case PORT_RESET:
if (hub.usb_port[n].device != NULL) {
hub.usb_port[n].device->usb_send_msg(USB_MSG_RESET);
hub.usb_port[n].PortChange |= PORT_STAT_C_RESET;
/* set enable bit */
hub.usb_port[n].PortStatus |= PORT_STAT_ENABLE;
hub.usb_port[n].PortStatus &= ~PORT_STAT_SUSPEND;
}
break;
case PORT_POWER:
#if USB_HUB_POWER_SWITCHING
#if USB_HUB_POWER_PER_PORT == 0
for (n = 0; n < hub.n_ports; n++) {
#endif
if ((hub.usb_port[n].PortStatus & PORT_STAT_POWER) == 0) {
hub.usb_port[n].PortStatus = PORT_STAT_POWER;
hub.device_change |= (1 << n);
runtime_config();
}
#if USB_HUB_POWER_PER_PORT == 0
}
#endif
#endif
break;
default:
BX_ERROR(("Unknown SetPortFeature: %d", value));
goto fail;
}
ret = 0;
break;
case OtherOutClassRequest | USB_REQ_CLEAR_FEATURE:
n = index - 1;
if (n >= hub.n_ports)
goto fail;
switch(value) {
case PORT_POWER:
#if USB_HUB_POWER_SWITCHING
#if USB_HUB_POWER_PER_PORT == 0
for (n = 0; n < hub.n_ports; n++) {
#endif
if (hub.usb_port[n].PortStatus & PORT_STAT_POWER) {
hub.usb_port[n].PortStatus &= ~PORT_STAT_POWER;
hub.device_change |= (1 << n);
runtime_config();
hub.usb_port[n].PortStatus = 0;
hub.usb_port[n].PortChange = 0;
}
#if USB_HUB_POWER_PER_PORT == 0
}
#endif
#endif
break;
case PORT_ENABLE:
hub.usb_port[n].PortStatus &= ~PORT_STAT_ENABLE;
break;
case PORT_C_ENABLE:
hub.usb_port[n].PortChange &= ~PORT_STAT_C_ENABLE;
break;
case PORT_SUSPEND:
hub.usb_port[n].PortStatus &= ~PORT_STAT_SUSPEND;
break;
case PORT_C_SUSPEND:
hub.usb_port[n].PortChange &= ~PORT_STAT_C_SUSPEND;
break;
case PORT_C_CONNECTION:
hub.usb_port[n].PortChange &= ~PORT_STAT_C_CONNECTION;
break;
case PORT_C_OVERCURRENT:
hub.usb_port[n].PortChange &= ~PORT_STAT_C_OVERCURRENT;
break;
case PORT_C_RESET:
hub.usb_port[n].PortChange &= ~PORT_STAT_C_RESET;
break;
default:
BX_ERROR(("Unknown ClearPortFeature: %d", value));
goto fail;
}
ret = 0;
break;
case DeviceClassInRequest | USB_REQ_GET_DESCRIPTOR:
if (
#if USB_HUB_VERSION == 0x0100
((value >> 8) == 0x00) || // USB 1.0 defines it as type = zero
#endif
((value >> 8) == 0x29)) { // USB 1.1+ defines it as type = 0x29
unsigned int var_hub_size = 0, indx = 7;
memcpy(data, bx_hub_hub_descriptor,
sizeof(bx_hub_hub_descriptor));
data[2] = hub.n_ports;
/* fill DeviceRemovable bits */
unsigned int byte_count = ((hub.n_ports + 1 + 7) / 8);
for (n = 0; n < byte_count; n++) {
data[indx + n] = 0x00;
var_hub_size++;
}
#if USB_HUB_VERSION == 0x0100
/* Build a bitmap of all present ports starting with:
bit 1 = port 0, bit 2 = port 1, etc.
We can assume there will be less than 16 ports (USB_HUB_MAX_PORTS < 16) */
unsigned int status = 0;
for (n = 0; n < hub.n_ports; n++) {
status |= (1 << (n + 1));
}
#endif
/* fill PortPwrCtrlMask bits */
/* USB 1.0: has this field set each bit indicating if a port is present
starting with bit 1. */
/* USB 1.1+: states that this field should be all 1's */
indx += var_hub_size;
for (n = 0; n < byte_count; n++) {
#if USB_HUB_VERSION == 0x0100
#if USB_HUB_POWER_SWITCHING && USB_HUB_POWER_PER_PORT
data[indx + n] = (Bit8u) (status & 0xFF);
status >>= 8;
#else
data[indx + n] = 0x00;
#endif
#else
data[indx + n] = 0xFF;
#endif
var_hub_size++;
}
ret = sizeof(bx_hub_hub_descriptor) + var_hub_size;
data[0] = ret;
#if USB_HUB_VERSION != 0x0100
} else if ((value >> 8) == 0x00) { // USB 1.0 defines it as type = zero
BX_INFO(("handle_control: Hub Class: A request of zero is a USB 1.0 request. USB 1.1+ use 0x29."));
goto fail;
#endif
} else {
BX_ERROR(("handle_control: Hub Class: unknown type requested: 0x%02x", value >> 8));
goto fail;
}
break;
default:
BX_ERROR(("handle_control: unknown request: 0x%04x", request));
fail:
d.stall = 1;
ret = USB_RET_STALL;
break;
}
return ret;
}
int usb_hub_device_c::handle_data(USBPacket *p)
{
int ret = 0;
// check that the length is <= the max packet size of the device
if (p->len > get_mps(p->devep)) {
BX_DEBUG(("EP%d transfer length (%d) is greater than Max Packet Size (%d).", p->devep, p->len, get_mps(p->devep)));
}
switch(p->pid) {
case USB_TOKEN_IN:
if (p->devep == 1) {
int i, n;
n = (hub.n_ports + 1 + 7) / 8;
if (p->len == 1) { /* FreeBSD workaround */
n = 1;
} else if (n > p->len) {
return USB_RET_BABBLE;
}
Bit16u status = 0;
for(i = 0; i < hub.n_ports; i++) {
if (hub.usb_port[i].PortChange)
status |= (1 << (i + 1));
}
if (status != hub.PortStatusC) {
hub.PortStatusC = status;
status |= 1; // bit 0 = hub change detected
}
#if USB_HUB_ALWAYS_REPORT != 1
if (status != 0) {
#endif
for(i = 0; i < n; i++) {
p->data[i] = status >> (8 * i);
}
ret = n;
#if USB_HUB_ALWAYS_REPORT != 1
} else {
ret = USB_RET_NAK;
}
#endif
} else {
goto fail;
}
break;
case USB_TOKEN_OUT:
default:
fail:
d.stall = 1;
ret = USB_RET_STALL;
break;
}
return ret;
}
int usb_hub_device_c::broadcast_packet(USBPacket *p)
{
int i, ret;
usb_device_c *dev;
for(i = 0; i < hub.n_ports; i++) {
dev = hub.usb_port[i].device;
if ((dev != NULL) && (hub.usb_port[i].PortStatus & PORT_STAT_ENABLE)) {
ret = dev->handle_packet(p);
if (ret != USB_RET_NODEV) {
return ret;
}
}
}
return USB_RET_NODEV;
}
int usb_hub_device_c::handle_packet(USBPacket *p)
{
if ((d.state >= USB_STATE_DEFAULT) &&
(d.addr != 0) &&
(p->devaddr != d.addr) &&
((p->pid == USB_TOKEN_SETUP) ||
(p->pid == USB_TOKEN_OUT) ||
(p->pid == USB_TOKEN_IN))) {
/* broadcast the packet to the devices */
return broadcast_packet(p);
}
return usb_device_c::handle_packet(p);
}
int hub_event_handler(int event, void *ptr, void *dev, int port);
void usb_hub_device_c::init_device(Bit8u port, bx_list_c *portconf)
{
char pname[BX_PATHNAME_LEN];
if (DEV_usb_init_device(portconf, this, &hub.usb_port[port].device, hub_event_handler, port)) {
if (usb_set_connect_status(port, 1)) {
portconf->get_by_name("options")->set_enabled(0);
sprintf(pname, "port%d.device", port+1);
bx_list_c *sr_list = (bx_list_c *) SIM->get_param(pname, hub.state);
hub.usb_port[port].device->register_state(sr_list);
}
}
}
void usb_hub_device_c::remove_device(Bit8u port)
{
if (hub.usb_port[port].device != NULL) {
delete hub.usb_port[port].device;
hub.usb_port[port].device = NULL;
}
}
int hub_event_handler(int event, void *ptr, void *dev, int port)
{
if (dev != NULL) {
return ((usb_hub_device_c *) dev)->event_handler(event, ptr, port);
}
return -1;
}
int usb_hub_device_c::event_handler(int event, void *ptr, int port)
{
int ret = 0;
switch (event) {
// packet events start here
case USB_EVENT_WAKEUP:
if (hub.usb_port[port].PortStatus & PORT_STAT_SUSPEND) {
hub.usb_port[port].PortChange |= PORT_STAT_C_SUSPEND;
}
if (d.event.dev != NULL) {
d.event.cb(USB_EVENT_WAKEUP, NULL, d.event.dev, d.event.port);
}
break;
// "host controller" events start here
case USB_EVENT_DEFAULT_SPEED:
// return default speed for specified port
return USB_SPEED_FULL;
case USB_EVENT_CHECK_SPEED:
if (ptr != NULL) {
usb_device_c *usb_device = (usb_device_c *) ptr;
if (usb_device->get_speed() <= d.speed)
ret = 1;
}
break;
default:
BX_ERROR(("unknown/unsupported event (id=%d) on port #%d", event, port+1));
ret = -1; // unknown event, event not handled
}
return ret;
}
bool usb_hub_device_c::usb_set_connect_status(Bit8u port, bool connected)
{
usb_device_c *device = hub.usb_port[port].device;
int hubnum = atoi(hub.config->get_name() + 6);
if (device != NULL) {
if (connected) {
switch (device->get_speed()) {
case USB_SPEED_LOW:
hub.usb_port[port].PortStatus |= PORT_STAT_LOW_SPEED;
break;
case USB_SPEED_FULL:
hub.usb_port[port].PortStatus &= ~PORT_STAT_LOW_SPEED;
break;
case USB_SPEED_HIGH:
case USB_SPEED_SUPER:
BX_PANIC(("Hub supports 'low' or 'full' speed devices only."));
usb_set_connect_status(port, 0);
return 0;
default:
BX_PANIC(("USB device returned invalid speed value"));
usb_set_connect_status(port, 0);
return 0;
}
hub.usb_port[port].PortStatus |= PORT_STAT_CONNECTION;
hub.usb_port[port].PortChange |= PORT_STAT_C_CONNECTION;
if (hub.usb_port[port].PortStatus & PORT_STAT_SUSPEND) {
hub.usb_port[port].PortChange |= PORT_STAT_C_SUSPEND;
}
if (d.event.dev != NULL) {
d.event.cb(USB_EVENT_WAKEUP, NULL, d.event.dev, d.event.port);
}
if (!device->get_connected()) {
if (!device->init()) {
usb_set_connect_status(port, 0);
BX_ERROR(("hub #%d, port #%d: connect failed", hubnum, port+1));
return 0;
} else {
BX_INFO(("hub #%d, port #%d: connect: %s", hubnum, port+1, device->get_info()));
}
}
} else {
BX_INFO(("hub #%d, port #%d: device disconnect", hubnum, port+1));
if (d.event.dev != NULL) {
d.event.cb(USB_EVENT_WAKEUP, NULL, d.event.dev, d.event.port);
}
hub.usb_port[port].PortStatus &= ~PORT_STAT_CONNECTION;
hub.usb_port[port].PortChange |= PORT_STAT_C_CONNECTION;
if (hub.usb_port[port].PortStatus & PORT_STAT_ENABLE) {
hub.usb_port[port].PortStatus &= ~PORT_STAT_ENABLE;
hub.usb_port[port].PortChange |= PORT_STAT_C_ENABLE;
}
hub.usb_port[port].PortStatus &= ~PORT_STAT_SUSPEND;
remove_device(port);
}
}
return connected;
}
void usb_hub_device_c::runtime_config()
{
int i;
char pname[6];
for (i = 0; i < hub.n_ports; i++) {
// device change support
if ((hub.device_change & (1 << i)) != 0) {
if (hub.usb_port[i].PortStatus & PORT_STAT_POWER) {
if ((hub.usb_port[i].PortStatus & PORT_STAT_CONNECTION) == 0) {
sprintf(pname, "port%d", i + 1);
init_device(i, (bx_list_c *) SIM->get_param(pname, hub.config));
} else {
usb_set_connect_status(i, 0);
}
hub.device_change &= ~(1 << i);
} else {
usb_set_connect_status(i, 0);
}
}
// forward to connected device
if (hub.usb_port[i].device != NULL) {
hub.usb_port[i].device->runtime_config();
}
}
}
#undef LOG_THIS
#define LOG_THIS hub->
// USB hub runtime parameter handler
Bit64s usb_hub_device_c::hub_param_handler(bx_param_c *param, bool set, Bit64s val)
{
int portnum;
usb_hub_device_c *hub;
bx_list_c *port;
if (set) {
port = (bx_list_c *) param->get_parent();
hub = (usb_hub_device_c *) (port->get_parent()->get_device_param());
if (hub != NULL) {
portnum = atoi(port->get_name()+4) - 1;
bool empty = (val == 0);
if ((portnum >= 0) && (portnum < hub->hub.n_ports)) {
if (empty && (hub->hub.usb_port[portnum].PortStatus & PORT_STAT_CONNECTION)) {
hub->hub.device_change |= (1 << portnum);
} else if (!empty && !(hub->hub.usb_port[portnum].PortStatus & PORT_STAT_CONNECTION)) {
hub->hub.device_change |= (1 << portnum);
} else if (val != ((bx_param_enum_c *) param)->get()) {
BX_ERROR(("hub_param_handler(): port #%d already in use", portnum+1));
val = ((bx_param_enum_c *) param)->get();
}
} else {
BX_PANIC(("usb_param_handler called with unexpected parameter '%s'", param->get_name()));
}
// hub == NULL. We shouldn't call BX_PANIC with a NULL pointer.
// #define BX_PANIC(x) (LOG_THIS /* = hub-> */ panic) x
//} else {
// BX_PANIC(("hub_param_handler: external hub not found"));
}
}
return val;
}
// USB runtime parameter enable handler
bool usb_hub_device_c::hub_param_enable_handler(bx_param_c *param, bool en)
{
bx_list_c *port = (bx_list_c *) param->get_parent();
int portnum = atoi(port->get_name() + 4) - 1;
usb_hub_device_c *hub = (usb_hub_device_c *)(port->get_parent()->get_device_param());
if (hub != NULL) {
if (en && (hub->hub.usb_port[portnum].device != NULL)) {
en = 0;
}
}
return en;
}
// USB runtime parameter handler: over-current
Bit64s usb_hub_device_c::hub_param_oc_handler(bx_param_c *param, bool set, Bit64s val)
{
int portnum;
usb_hub_device_c *hub;
bx_list_c *port;
if (set && val) {
port = (bx_list_c *) param->get_parent();
hub = (usb_hub_device_c *) (port->get_parent()->get_device_param());
if (hub != NULL) {
portnum = atoi(port->get_name()+4) - 1;
#if USB_HUB_POWER_SWITCHING
hub->hub.usb_port[portnum].PortStatus &= ~PORT_STAT_POWER;
#endif
hub->hub.usb_port[portnum].PortStatus |= PORT_STAT_OVERCURRENT;
hub->hub.usb_port[portnum].PortChange |= PORT_STAT_C_OVERCURRENT;
BX_DEBUG(("Over-current signaled on port #%d.", portnum + 1));
}
}
return 0; // clear the indicator for next time
}
void usb_hub_restore_handler(void *dev, bx_list_c *conf)
{
if (dev != NULL) {
((usb_hub_device_c *) dev)->restore_handler(conf);
}
}
void usb_hub_device_c::restore_handler(bx_list_c *conf)
{
int i;
const char *pname = conf->get_name();
i = atoi(&pname[4]) - 1;
init_device(i, (bx_list_c *) SIM->get_param(pname, hub.config));
}
usb_device_c *usb_hub_device_c::find_device(Bit8u addr)
{
int i;
usb_device_c *dev, *dev2;
if (addr == d.addr) {
return this;
} else {
for (i = 0; i < hub.n_ports; i++) {
dev = hub.usb_port[i].device;
if ((dev != NULL) && (hub.usb_port[i].PortStatus & PORT_STAT_ENABLE)) {
dev2 = dev->find_device(addr);
if (dev2 != NULL) {
return dev2;
}
}
}
return NULL;
}
}
#endif // BX_SUPPORT_PCI && BX_SUPPORT_PCIUSB