Bochs/bochs/iodev/usb/usb_xhci.cc
Volker Ruppert e00a5d0761 Started rewrite of the USB devices handling.
- Moved main option from the portX parameter to the optionsX parameter. Added
  new option "path" for the disk, cdrom and floppy devices, "ports" for the
  USB hub and "file" for the USB printer. The legacy syntax is still supported.
- Register USB device state after init() to ensure all options that may have
  effect on the save/restore list are applied.
- Report available devices on top of log file similar to hdimage, network and
  sound modules.
- Changed USB device type variable typein host controllers to int.
- Prepared improved usbdev_ctl init / exit implementation.
- TODO: Create list of USB devices dynamicly from available plugins.
2021-02-14 08:30:49 +00:00

3305 lines
154 KiB
C++

/////////////////////////////////////////////////////////////////////////
// $Id$
/////////////////////////////////////////////////////////////////////////
//
// Copyright (C) 2010-2017 Benjamin D Lunt (fys [at] fysnet [dot] net)
// 2011-2021 The Bochs Project
//
// This library 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 of the License, or (at your option) any later version.
//
// This library 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., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
/////////////////////////////////////////////////////////////////////////
// Experimental USB xHCI adapter
// 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
/* Notes by Ben Lunt:
* This emulates a NEC/Renesas uPD720202 2-port (2 socket, 4-port register sets)
* Root Hub xHCI Host Controller.
* Many, many thanks to Renesas for their work and effort in helping my research.
*
* I have tested in with my own tests and WinXP Home Edition SP3.
*
* Use port1 and port2 to emulate a Super-speed device, and use port3 and port4
* to emulate Low-, Full-, or High-speed devices.
*
* Examples:
* usb_xhci: enabled=1, port3=disk:"usbdisk.img", options3=speed:low
* usb_xhci: enabled=1, port3=disk:"usbdisk.img", options3=speed:full
* usb_xhci: enabled=1, port3=disk:"usbdisk.img", options3=speed:high
* usb_xhci: enabled=1, port1=disk:"usbdisk.img", options1=speed:super
*
* Currently, only a USB MSD Thumb Drive device is emulated as a Super-speed device.
* All other emulated devices are Low- or Full-speed devices.
*
* This code is 32- and 64-bit addressing aware (uses: bx_phy_address & ADDR_CAP_64)
*
* This emulation is coded with the new registers of xHCI version 1.10. However,
* nothing more has been done to emulate 1.10 features. YET.
* This emulation is set with VERSION_MAJOR and VERSION_MINOR, 0x01 and 0x10
* respectively.
*
* This emulation is experimental. It is not completly accorate, though it is
* my intention to make it as close as possible. If you find an error or would
* like to add to the emulation, please let me know at fys [at] fysnet [dot] net.
*
*
*/
#include "iodev.h"
#if BX_SUPPORT_PCI && BX_SUPPORT_USB_XHCI
#include "pci.h"
#include "usb_common.h"
#include "usb_xhci.h"
#define LOG_THIS theUSB_XHCI->
bx_usb_xhci_c* theUSB_XHCI = NULL;
Bit8u port_speed_allowed[USB_XHCI_PORTS] = { USB3, USB3, USB2, USB2 };
Bit8u ext_caps[EXT_CAPS_SIZE] = {
/* 0x500 */ 0x01, // Legacy Support
0x04, // next = 0x04 * 4 = 0x510
0x00, 0x00, // Semaphores
0x00, 0x00, 0x00, 0xE0, // SMI on Bar, SMI on Command, SMI on Semaphore
0x00, 0x00, 0x00, 0x00, // reserved
0x00, 0x00, 0x00, 0x00, // filler
/* 0x510 */ 0x02, // Port Protocol
0x05, // next = 0x05 * 4 = 0x524
0x00, 0x03, // Version 3.0
0x55, 0x53, 0x42, 0x20, // "USB "
0x01, // 1-based starting index
0x02, // count of 2 port registers starting at base above
0x00, 0x00, // reserved
0x00, 0x00, 0x00, 0x00, // reserved
0x00, 0x00, 0x00, 0x00, // filler
/* 0x524 */ 0x02, // Port Protocol
0x07, // next = 0x07 * 4 = 0x540
0x00, 0x02, // Version 2.0
0x55, 0x53, 0x42, 0x20, // "USB "
0x03, // 1-based starting index
0x02, // count of 2 port registers starting at base above
0x00, 0x00, //
0x00, 0x00, 0x00, 0x00, //
0x00, 0x00, 0x00, 0x00, //
0x00, 0x00, 0x00, 0x00, //
0x00, 0x00, 0x00, 0x00, //
/* The following two items are not needed or used by this emulation, but they
* are present in the uPD720202 controller. I leave them here to closely emulate
* the said controller, and with intentions of some day adding the debug support.
* If you wish to comment them out, please be sure to remember to change the
* 0x07, // (next = 0x07 * 4 = 0x540)
* above to a
* 0x00, // no more items
*/
/* 0x540 */ 0xC0, // Vendor Defined
0x04, // next = 0x04 * 4 = 0x550
0x00, 0x00, //
0x00, 0x00, 0x00, 0x00, //
0x00, 0x00, 0x00, 0x00, //
0x00, 0x00, 0x00, 0x00, //
/* 0x550 */ 0x0A, // Debug Registers
0x00, // no more
0x00, 0x00, //
0x00, 0x00, 0x00, 0x00, // Doorbell Target
0x00, 0x00, 0x00, 0x00, // Event Ring Table Size
0x00, 0x00, 0x00, 0x00, // Event Ring Table Base Address
0x00, 0x00, 0x00, 0x00, // (64-bit)
0x00, 0x00, 0x00, 0x00, // Dequeue Pointer
0x00, 0x00, 0x00, 0x00, // (64-bit)
0x00, 0x00, 0x00, 0x00, // Max Burst Size, Dev Address, etc.
0x00, 0x00, 0x00, 0x00, // Debug Port Number
0x00, 0x00, 0x00, 0x00, // Port Status
0x00, 0x00, 0x00, 0x00, // Context Pointer
0x00, 0x00, 0x00, 0x00, // (64-bit)
0x00, 0x00, 0x00, 0x00, // DbC Protocol, Vendor ID
0x00, 0x00, 0x00, 0x00, // Product ID, Device Revision
0x00, 0x00, 0x00, 0x00, // filler
0x00, 0x00, 0x00, 0x00 // filler
};
/* See Section 6.2.6 of xHCI version 1.00
* Bandwidth of each port (4) of each speed.
*/
Bit8u port_band_width[4][1 + USB_XHCI_PORTS] = {
/* rsvd, port1, port2, port3, port4 */
/* Full Speed */ { 0x00, 0, 0, 90, 90 },
/* Low Speed */ { 0x00, 0, 0, 90, 90 },
/* High Speed */ { 0x00, 0, 0, 80, 80 },
/* Super Speed */ { 0x00, 90, 90, 0, 0 }
};
// builtin configuration handling functions
Bit32s usb_xhci_options_parser(const char *context, int num_params, char *params[])
{
if (!strcmp(params[0], "usb_xhci")) {
bx_list_c *base = (bx_list_c*) SIM->get_param(BXPN_USB_XHCI);
for (int i = 1; i < num_params; i++) {
if (!strncmp(params[i], "enabled=", 8)) {
SIM->get_param_bool(BXPN_XHCI_ENABLED)->set(atol(&params[i][8]));
} else if (!strncmp(params[i], "port", 4)) {
if (SIM->parse_usb_port_params(context, 0, params[i], BX_N_USB_XHCI_PORTS, base) < 0) {
return -1;
}
} else if (!strncmp(params[i], "options", 7)) {
if (SIM->parse_usb_port_params(context, 1, params[i], BX_N_USB_XHCI_PORTS, base) < 0) {
return -1;
}
} else {
BX_ERROR(("%s: unknown parameter '%s' for usb_xhci ignored.", context, params[i]));
}
}
} else {
BX_PANIC(("%s: unknown directive '%s'", context, params[0]));
}
return 0;
}
Bit32s usb_xhci_options_save(FILE *fp)
{
bx_list_c *base = (bx_list_c*) SIM->get_param(BXPN_USB_XHCI);
SIM->write_usb_options(fp, BX_N_USB_XHCI_PORTS, base);
return 0;
}
// device plugin entry point
PLUGIN_ENTRY_FOR_MODULE(usb_xhci)
{
if (mode == PLUGIN_INIT) {
theUSB_XHCI = new bx_usb_xhci_c();
BX_REGISTER_DEVICE_DEVMODEL(plugin, type, theUSB_XHCI, BX_PLUGIN_USB_XHCI);
// add new configuration parameter for the config interface
SIM->init_usb_options("xHCI", "xhci", BX_N_USB_XHCI_PORTS);
// register add-on option for bochsrc and command line
SIM->register_addon_option("usb_xhci", usb_xhci_options_parser, usb_xhci_options_save);
} else if (mode == PLUGIN_FINI) {
SIM->unregister_addon_option("usb_xhci");
bx_list_c *menu = (bx_list_c*)SIM->get_param("ports.usb");
delete theUSB_XHCI;
menu->remove("xhci");
} else {
return (int)PLUGTYPE_OPTIONAL;
}
return 0; // Success
}
// the device object
bx_usb_xhci_c::bx_usb_xhci_c()
{
put("usb_xhci", "XHCI");
memset((void*)&hub, 0, sizeof(bx_usb_xhci_t));
rt_conf_id = -1;
xhci_timer_index = BX_NULL_TIMER_HANDLE;
}
bx_usb_xhci_c::~bx_usb_xhci_c()
{
char pname[16];
SIM->unregister_runtime_config_handler(rt_conf_id);
for (int i=0; i<USB_XHCI_PORTS; i++) {
sprintf(pname, "port%d.device", i+1);
SIM->get_param_string(pname, SIM->get_param(BXPN_USB_XHCI))->set_handler(NULL);
remove_device(i);
}
SIM->get_bochs_root()->remove("usb_xhci");
bx_list_c *usb_rt = (bx_list_c*)SIM->get_param(BXPN_MENU_RUNTIME_USB);
usb_rt->remove("xhci");
BX_DEBUG(("Exit"));
}
void bx_usb_xhci_c::init(void)
{
unsigned i;
char pname[6];
bx_list_c *xhci, *port;
bx_param_string_c *device;
/* If you wish to set DEBUG=report in the code, instead of
* in the configuration, simply uncomment this line. I use
* it when I am working on this emulation.
*/
//LOG_THIS setonoff(LOGLEV_DEBUG, ACT_REPORT);
// Read in values from config interface
xhci = (bx_list_c*) SIM->get_param(BXPN_USB_XHCI);
// Check if the device is disabled or not configured
if (!SIM->get_param_bool("enabled", xhci)->get()) {
BX_INFO(("USB xHCI disabled"));
// mark unused plugin for removal
((bx_param_bool_c*)((bx_list_c*)SIM->get_param(BXPN_PLUGIN_CTRL))->get_by_name("usb_xhci"))->set(0);
return;
}
BX_XHCI_THIS xhci_timer_index =
DEV_register_timer(this, xhci_timer_handler, 1024, 1, 1, "xhci_timer");
BX_XHCI_THIS devfunc = 0x00;
DEV_register_pci_handlers(this, &BX_XHCI_THIS devfunc, BX_PLUGIN_USB_XHCI,
"Experimental USB xHCI");
// initialize readonly registers
// 0x1912 = vendor (Renesas)
// 0x0015 = device (0x0014 = uPD720201, 0x0015 = uPD720202)
// revision number (0x03 = uPD720201, 0x02 = uPD720202)
init_pci_conf(0x1912, 0x0015, 0x02, 0x0c0330, 0x00, BX_PCI_INTD);
BX_XHCI_THIS init_bar_mem(0, IO_SPACE_SIZE, read_handler, write_handler);
// initialize capability registers
BX_XHCI_THIS hub.cap_regs.HcCapLength = (VERSION_MAJOR << 24) | (VERSION_MINOR << 16) | OPS_REGS_OFFSET;
BX_XHCI_THIS hub.cap_regs.HcSParams1 = (USB_XHCI_PORTS << 24) | (INTERRUPTERS << 8) | MAX_SLOTS;
BX_XHCI_THIS hub.cap_regs.HcSParams2 =
// MAX_SCRATCH_PADS 4:0 in bits 31:27 (v1.00) and bits 9:5 in bits 21:25 (v1.01+)
((MAX_SCRATCH_PADS >> 5) << 21) | ((MAX_SCRATCH_PADS & 0x1F) << 27) |
((SCATCH_PAD_RESTORE == 1) << 26) | (MAX_SEG_TBL_SZ_EXP << 4) | ISO_SECH_THRESHOLD;
BX_XHCI_THIS hub.cap_regs.HcSParams3 = (U2_DEVICE_EXIT_LAT << 16) | U1_DEVICE_EXIT_LAT;
BX_XHCI_THIS hub.cap_regs.HcCParams1 =
((EXT_CAPS_OFFSET >> 2) << 16) | (MAX_PSA_SIZE << 12) | (SEC_DOMAIN_BAND << 9) | (PARSE_ALL_EVENT << 8) |
(LIGHT_HC_RESET << 5) |
(PORT_INDICATORS << 4) | (PORT_POWER_CTRL << 3) | ((CONTEXT_SIZE >> 6) << 2) |
(BW_NEGOTIATION << 1) | ADDR_CAP_64;
BX_XHCI_THIS hub.cap_regs.DBOFF = DOORBELL_OFFSET; // at offset DOORBELL_OFFSET from base
BX_XHCI_THIS hub.cap_regs.RTSOFF = RUNTIME_OFFSET; // at offset RUNTIME_OFFSET from base
#if ((VERSION_MAJOR == 1) && (VERSION_MINOR >= 0x10))
BX_XHCI_THIS hub.cap_regs.HcCParams2 =
(CONFIG_INFO_CAP << 5) | (LARGE_ESIT_PAYLOAD_CAP << 4) | (COMPLNC_TRANS_CAP << 3) |
(FORCE_SAVE_CONTEXT_CAP << 2) | (CONFIG_EP_CMND_CAP << 1) | U3_ENTRY_CAP;
#endif
// initialize runtime configuration
bx_list_c *usb_rt = (bx_list_c*)SIM->get_param(BXPN_MENU_RUNTIME_USB);
bx_list_c *xhci_rt = new bx_list_c(usb_rt, "xhci", "xHCI Runtime Options");
xhci_rt->set_options(xhci_rt->SHOW_PARENT | xhci_rt->USE_BOX_TITLE);
for (i=0; i<USB_XHCI_PORTS; i++) {
sprintf(pname, "port%d", i+1);
port = (bx_list_c*)SIM->get_param(pname, xhci);
xhci_rt->add(port);
device = (bx_param_string_c*)port->get_by_name("device");
device->set_handler(usb_param_handler);
BX_XHCI_THIS hub.usb_port[i].device = NULL;
BX_XHCI_THIS hub.usb_port[i].portsc.ccs = 0;
BX_XHCI_THIS hub.usb_port[i].portsc.csc = 0;
}
// register handler for correct device connect handling after runtime config
BX_XHCI_THIS rt_conf_id = SIM->register_runtime_config_handler(BX_XHCI_THIS_PTR, runtime_config_handler);
BX_XHCI_THIS device_change = 0;
BX_XHCI_THIS packets = NULL;
for (i=0; i<USB_XHCI_PORTS; i++)
BX_XHCI_THIS hub.usb_port[i].is_usb3 = (port_speed_allowed[i] == USB3);
BX_INFO(("USB xHCI initialized"));
}
void bx_usb_xhci_c::reset(unsigned type)
{
unsigned i;
if (type == BX_RESET_HARDWARE) {
static const struct reset_vals_t {
unsigned addr;
unsigned char val;
} reset_vals[] = {
{ 0x04, 0x06 }, { 0x05, 0x01 }, // command_io
{ 0x06, 0x10 }, { 0x07, 0x00 }, // status (has caps list)
{ 0x0C, 0x10 }, // cache line size
{ 0x0D, 0x00 }, // bus latency
{ 0x0F, 0x00 }, // BIST is not supported
// address space 0x10 - 0x17
{ 0x10, 0x04 }, { 0x11, 0x00 }, // 64-bit wide and anywhere in the 64-bit address space
{ 0x12, 0x50 }, { 0x13, 0xF0 }, //
{ 0x14, 0x00 }, { 0x15, 0x00 }, //
{ 0x16, 0x00 }, { 0x17, 0x00 }, //
// addresses 0x18 - 0x28 are reserved
{ 0x2C, 0xFF }, { 0x2D, 0xFF }, // subsystem vendor ID
{ 0x2E, 0xFF }, { 0x2F, 0xFF }, // subsystem ID
{ 0x34, 0x50 }, // offset of capabilities list within configuration space
{ 0x3C, 0x0A }, // IRQ
{ 0x3E, 0x00 }, // minimum time bus master needs PCI bus ownership, in 250ns units
{ 0x3F, 0x00 }, // maximum latency, in 250ns units (bus masters only) (read-only)
// capabilities list:
{ 0x50, 0x01 }, // PCI Power Management
// { 0x51, 0x70 }, // Pointer to next item (0x00 = no more)
{ 0x51, 0x00 }, // Pointer to next item (0x00 = no more)
{ 0x52, 0xC3 }, { 0x53, 0xC9 }, // Capabilities: version = 1.2, Aux Current = 375mA,
{ 0x54, 0x08 }, { 0x55, 0x00 }, // Status: Power State = D0, Bit 3 = no soft reset
{ 0x56, 0x00 }, { 0x57, 0x00 }, //
{ 0x60, 0x30 }, // Supports USB Spec version 3.0
{ 0x61, 0x20 }, // Frame List Adjustment
// Firmware version
{ 0x6C, 0x09 }, // ????? This byte not documented in specification ??
{ 0x6D, 0x18 }, { 0x6E, 0x20 }, // low = 0x18, high = 0x20
{ 0x6F, 0x00 }, //
/* The following items are used with PCIe and are not wanted or needed within
* this emulation. However, the controller contains them within its PCI(e)
* configuration space. Therefore, I leave them here (though commented out)
* for that sake. This is for the benefit of the reader, along as myself.
* If you uncomment this section, remember to swap the commented lines above.
* { 0x51, 0x70 }, and { 0x51, 0x00 },
// MSI
{ 0x70, 0x05 }, // MSI
{ 0x71, 0x90 }, // Pointer to next item (0x00 = no more)
{ 0x72, 0x86 }, { 0x73, 0x00 }, //
{ 0x74, 0x00 }, { 0x75, 0x00 }, // 0x74 - 0x87 are zero's
// MSI-x
{ 0x90, 0x11 }, // MSI-x
{ 0x91, 0xA0 }, // Pointer to next item (0x00 = no more)
{ 0x92, 0x07 }, { 0x93, 0x00 }, //
{ 0x94, 0x00 }, { 0x95, 0x10 }, //
{ 0x96, 0x00 }, { 0x97, 0x00 }, //
{ 0x98, 0x80 }, { 0x99, 0x10 }, //
{ 0x9A, 0x00 }, { 0x9B, 0x00 }, //
// PCIe caps register1
{ 0xA0, 0x10 }, // PCIe caps register
{ 0xA1, 0x00 }, // Pointer to next item (0x00 = no more)
{ 0xA2, 0x02 }, { 0xA3, 0x00 }, //
{ 0xA4, 0xC0 }, { 0xA5, 0x8F }, //
{ 0xA6, 0x00 }, { 0xA7, 0x00 }, //
{ 0xA8, 0x10 }, { 0xA9, 0x28 }, //
{ 0xAA, 0x10 }, { 0xAB, 0x00 }, //
{ 0xAC, 0x12 }, { 0xAD, 0xEC }, //
{ 0xAE, 0x07 }, { 0xAF, 0x00 }, //
{ 0xB0, 0x40 }, { 0xB1, 0x00 }, //
{ 0xB2, 0x11 }, { 0xB3, 0x10 }, //
// PCIe caps register2
{ 0xC4, 0x10 }, { 0xC5, 0x08 }, //
{ 0xC6, 0x00 }, { 0xC7, 0x00 }, //
{ 0xC8, 0x00 }, { 0xC9, 0x00 }, //
{ 0xCA, 0x00 }, { 0xCB, 0x00 }, //
{ 0xCC, 0x00 }, { 0xCD, 0x00 }, //
{ 0xCE, 0x00 }, { 0xCF, 0x00 }, //
{ 0xD0, 0x00 }, { 0xD1, 0x00 }, //
{ 0xD2, 0x01 }, { 0xD3, 0x10 }, //
// Phy 0
{ 0xDC, 0x00 }, { 0xDD, 0x00 }, //
{ 0xDE, 0x00 }, { 0xDF, 0x00 }, //
// Phy 1
{ 0xE0, 0x00 }, { 0xE1, 0x00 }, //
{ 0xE2, 0x00 }, { 0xE3, 0x00 }, //
// Phy 2 (battery charging control)
{ 0xE4, 0x00 }, { 0xE5, 0x00 }, //
{ 0xE6, 0x03 }, { 0xE7, 0x00 }, //
// HCConfiguration Register
{ 0xE8, 0x00 }, { 0xE9, 0x00 }, //
{ 0xEA, 0x01 }, { 0xEB, 0x05 }, //
// External ROM Information Register
{ 0xEC, 0x7F }, { 0xED, 0x20 }, //
{ 0xEE, 0x9D }, { 0xEF, 0x01 }, //
// External ROM Information Register
{ 0xF0, 0x00 }, { 0xF1, 0x00 }, //
{ 0xF2, 0x00 }, { 0xF3, 0x00 }, //
// Firmware Download Control and Status
{ 0xF4, 0x00 }, { 0xF5, 0x00 }, //
{ 0xF6, 0x00 }, { 0xF7, 0x80 }, //
// Data 0
{ 0xF8, 0x00 }, { 0xF9, 0x00 }, //
{ 0xFA, 0x00 }, { 0xFB, 0x00 }, //
// Data 1
{ 0xFC, 0x00 }, { 0xFD, 0x00 }, //
{ 0xFE, 0x00 }, { 0xFF, 0x00 }, //
* Offsets 0x100 -> 0x157 are used by controller,
* but Bochs only supports up to 256 (PCI)
*/
};
for (i = 0; i < sizeof(reset_vals) / sizeof(*reset_vals); i++) {
BX_XHCI_THIS pci_conf[reset_vals[i].addr] = reset_vals[i].val;
}
}
BX_XHCI_THIS reset_hc();
}
void bx_usb_xhci_c::reset_hc()
{
int i;
char pname[6];
// Command
BX_XHCI_THIS hub.op_regs.HcCommand.RsvdP1 = 0;
#if ((VERSION_MAJOR == 1) && (VERSION_MINOR >= 0x10))
BX_XHCI_THIS hub.op_regs.HcCommand.cme = 0;
BX_XHCI_THIS hub.op_regs.HcCommand.spe = 0;
#endif
BX_XHCI_THIS hub.op_regs.HcCommand.eu3s = 0;
BX_XHCI_THIS hub.op_regs.HcCommand.ewe = 0;
BX_XHCI_THIS hub.op_regs.HcCommand.crs = 0;
BX_XHCI_THIS hub.op_regs.HcCommand.css = 0;
BX_XHCI_THIS hub.op_regs.HcCommand.lhcrst = 0;
BX_XHCI_THIS hub.op_regs.HcCommand.RsvdP0 = 0;
BX_XHCI_THIS hub.op_regs.HcCommand.hsee = 0;
BX_XHCI_THIS hub.op_regs.HcCommand.inte = 0;
BX_XHCI_THIS hub.op_regs.HcCommand.hcrst = 0;
BX_XHCI_THIS hub.op_regs.HcCommand.rs = 0;
// Status
BX_XHCI_THIS hub.op_regs.HcStatus.RsvdZ1 = 0;
BX_XHCI_THIS hub.op_regs.HcStatus.hce = 0;
BX_XHCI_THIS hub.op_regs.HcStatus.cnr = 0;
BX_XHCI_THIS hub.op_regs.HcStatus.sre = 0;
BX_XHCI_THIS hub.op_regs.HcStatus.rss = 0;
BX_XHCI_THIS hub.op_regs.HcStatus.sss = 0;
BX_XHCI_THIS hub.op_regs.HcStatus.RsvdZ0 = 0;
BX_XHCI_THIS hub.op_regs.HcStatus.pcd = 0;
BX_XHCI_THIS hub.op_regs.HcStatus.eint = 0;
BX_XHCI_THIS hub.op_regs.HcStatus.hse = 0;
BX_XHCI_THIS hub.op_regs.HcStatus.RsvdZ2 = 0;
BX_XHCI_THIS hub.op_regs.HcStatus.hch = 1;
// Page Size
BX_XHCI_THIS hub.op_regs.HcPageSize.pagesize = XHCI_PAGE_SIZE;
// Device Notification Control Register
BX_XHCI_THIS hub.op_regs.HcNotification.RsvdP = 0;
BX_XHCI_THIS hub.op_regs.HcNotification.n15 = 0;
BX_XHCI_THIS hub.op_regs.HcNotification.n14 = 0;
BX_XHCI_THIS hub.op_regs.HcNotification.n13 = 0;
BX_XHCI_THIS hub.op_regs.HcNotification.n12 = 0;
BX_XHCI_THIS hub.op_regs.HcNotification.n11 = 0;
BX_XHCI_THIS hub.op_regs.HcNotification.n10 = 0;
BX_XHCI_THIS hub.op_regs.HcNotification.n9 = 0;
BX_XHCI_THIS hub.op_regs.HcNotification.n8 = 0;
BX_XHCI_THIS hub.op_regs.HcNotification.n7 = 0;
BX_XHCI_THIS hub.op_regs.HcNotification.n6 = 0;
BX_XHCI_THIS hub.op_regs.HcNotification.n5 = 0;
BX_XHCI_THIS hub.op_regs.HcNotification.n4 = 0;
BX_XHCI_THIS hub.op_regs.HcNotification.n3 = 0;
BX_XHCI_THIS hub.op_regs.HcNotification.n2 = 0;
BX_XHCI_THIS hub.op_regs.HcNotification.n1 = 0;
BX_XHCI_THIS hub.op_regs.HcNotification.n0 = 0;
// Command Ring Control Register
BX_XHCI_THIS hub.op_regs.HcCrcr.crc = 0;
BX_XHCI_THIS hub.op_regs.HcCrcr.RsvdP = 0;
BX_XHCI_THIS hub.op_regs.HcCrcr.crr = 0;
BX_XHCI_THIS hub.op_regs.HcCrcr.ca = 0;
BX_XHCI_THIS hub.op_regs.HcCrcr.cs = 0;
BX_XHCI_THIS hub.op_regs.HcCrcr.rcs = 0;
// DCBAAP
BX_XHCI_THIS hub.op_regs.HcDCBAAP.dcbaap = 0;
// Config
BX_XHCI_THIS hub.op_regs.HcConfig.RsvdP = 0;
#if ((VERSION_MAJOR == 1) && (VERSION_MINOR >= 0x10))
BX_XHCI_THIS hub.op_regs.HcConfig.u3e = 0;
BX_XHCI_THIS hub.op_regs.HcConfig.cie = 0;
#endif
BX_XHCI_THIS hub.op_regs.HcConfig.MaxSlotsEn = 0;
// Ports[x]
for (i=0; i<USB_XHCI_PORTS; i++) {
reset_port(i);
if (BX_XHCI_THIS hub.usb_port[i].device == NULL) {
sprintf(pname, "port%d", i+1);
init_device(i, (bx_list_c*)SIM->get_param(pname, SIM->get_param(BXPN_USB_XHCI)));
} else {
usb_set_connect_status(i, BX_XHCI_THIS hub.usb_port[i].device->get_type(), 1);
}
}
// Extended Caps
for (i=0; i<EXT_CAPS_SIZE; i++)
BX_XHCI_THIS hub.extended_caps[i] = ext_caps[i];
// interrupters[x]
BX_XHCI_THIS hub.runtime_regs.mfindex.RsvdP = 0;
BX_XHCI_THIS hub.runtime_regs.mfindex.index = 0;
for (i=0; i<INTERRUPTERS; i++) {
BX_XHCI_THIS hub.runtime_regs.interrupter[i].iman.RsvdP = 0;
BX_XHCI_THIS hub.runtime_regs.interrupter[i].iman.ie = 0;
BX_XHCI_THIS hub.runtime_regs.interrupter[i].iman.ip = 0;
BX_XHCI_THIS hub.runtime_regs.interrupter[i].imod.imodc = 0;
BX_XHCI_THIS hub.runtime_regs.interrupter[i].imod.imodi = 4000;
BX_XHCI_THIS hub.runtime_regs.interrupter[i].erstsz.RsvdP = 0;
BX_XHCI_THIS hub.runtime_regs.interrupter[i].erstsz.erstabsize = 0;
BX_XHCI_THIS hub.runtime_regs.interrupter[i].RsvdP = 0;
BX_XHCI_THIS hub.runtime_regs.interrupter[i].erstba.erstabadd = 0;
BX_XHCI_THIS hub.runtime_regs.interrupter[i].erstba.RsvdP = 0;
BX_XHCI_THIS hub.runtime_regs.interrupter[i].erdp.eventadd = 0;
BX_XHCI_THIS hub.runtime_regs.interrupter[i].erdp.ehb = 0;
BX_XHCI_THIS hub.runtime_regs.interrupter[i].erdp.desi = 0;
}
// reset our slot contexts
for (i=0; i<MAX_SLOTS; i++)
BX_XHCI_THIS hub.slots[i].enabled = 0;
while (BX_XHCI_THIS packets != NULL) {
usb_cancel_packet(&BX_XHCI_THIS packets->packet);
remove_async_packet(&BX_XHCI_THIS packets, BX_XHCI_THIS packets);
}
}
void bx_usb_xhci_c::reset_port(int p)
{
BX_XHCI_THIS hub.usb_port[p].portsc.wpr = 0;
BX_XHCI_THIS hub.usb_port[p].portsc.dr = 0;
BX_XHCI_THIS hub.usb_port[p].portsc.woe = 0;
BX_XHCI_THIS hub.usb_port[p].portsc.wde = 0;
BX_XHCI_THIS hub.usb_port[p].portsc.wce = 0;
BX_XHCI_THIS hub.usb_port[p].portsc.cas = 0;
BX_XHCI_THIS hub.usb_port[p].portsc.cec = 0;
BX_XHCI_THIS hub.usb_port[p].portsc.plc = 0;
BX_XHCI_THIS hub.usb_port[p].portsc.prc = 0;
BX_XHCI_THIS hub.usb_port[p].portsc.occ = 0;
BX_XHCI_THIS hub.usb_port[p].portsc.wrc = 0;
BX_XHCI_THIS hub.usb_port[p].portsc.pec = 0;
BX_XHCI_THIS hub.usb_port[p].portsc.csc = 0;
BX_XHCI_THIS hub.usb_port[p].portsc.lws = 0;
BX_XHCI_THIS hub.usb_port[p].portsc.pic = 0;
BX_XHCI_THIS hub.usb_port[p].portsc.speed = 0;
BX_XHCI_THIS hub.usb_port[p].portsc.pp = 0;
BX_XHCI_THIS hub.usb_port[p].portsc.pls = PLS_U0; // always ready to go (for our sake anyway)
BX_XHCI_THIS hub.usb_port[p].portsc.pr = 0;
BX_XHCI_THIS hub.usb_port[p].portsc.oca = 0;
BX_XHCI_THIS hub.usb_port[p].portsc.ped = 0;
if (BX_XHCI_THIS hub.usb_port[p].is_usb3) {
BX_XHCI_THIS hub.usb_port[p].usb3.portpmsc.RsvdP = 0;
BX_XHCI_THIS hub.usb_port[p].usb3.portpmsc.fla = 0;
BX_XHCI_THIS hub.usb_port[p].usb3.portpmsc.u2timeout = 0;
BX_XHCI_THIS hub.usb_port[p].usb3.portpmsc.u1timeout = 0;
BX_XHCI_THIS hub.usb_port[p].usb3.portli.lec = 0;
} else {
BX_XHCI_THIS hub.usb_port[p].usb2.portpmsc.tmode = 0;
BX_XHCI_THIS hub.usb_port[p].usb2.portpmsc.RsvdP = 0;
BX_XHCI_THIS hub.usb_port[p].usb2.portpmsc.hle = 0;
BX_XHCI_THIS hub.usb_port[p].usb2.portpmsc.l1dslot = 0;
BX_XHCI_THIS hub.usb_port[p].usb2.portpmsc.hird = 0;
BX_XHCI_THIS hub.usb_port[p].usb2.portpmsc.rwe = 0;
BX_XHCI_THIS hub.usb_port[p].usb2.portpmsc.l1s = 0;
BX_XHCI_THIS hub.usb_port[p].usb2.portli.RsvdP = 0;
}
BX_XHCI_THIS hub.usb_port[p].has_been_reset = 0;
}
void bx_usb_xhci_c::reset_port_usb3(int port, const int reset_type)
{
BX_INFO(("Reset port #%i, type=%i", port + 1, reset_type));
BX_XHCI_THIS hub.usb_port[port].portsc.pr = 0;
BX_XHCI_THIS hub.usb_port[port].has_been_reset = 1;
if (BX_XHCI_THIS hub.usb_port[port].portsc.ccs) {
BX_XHCI_THIS hub.usb_port[port].portsc.prc = 1;
BX_XHCI_THIS hub.usb_port[port].portsc.pls = PLS_U0;
BX_XHCI_THIS hub.usb_port[port].portsc.ped = 1;
if (BX_XHCI_THIS hub.usb_port[port].device != NULL) {
BX_XHCI_THIS hub.usb_port[port].device->usb_send_msg(USB_MSG_RESET);
if (BX_XHCI_THIS hub.usb_port[port].is_usb3 && (reset_type == WARM_RESET))
BX_XHCI_THIS hub.usb_port[port].portsc.wrc = 1;
BX_XHCI_THIS hub.usb_port[port].portsc.prc = 1;
}
} else {
BX_XHCI_THIS hub.usb_port[port].portsc.pls = PLS_RXDETECT;
BX_XHCI_THIS hub.usb_port[port].portsc.ped = 0;
BX_XHCI_THIS hub.usb_port[port].portsc.speed = 0;
}
}
/* This is the Save/Restore part of the controller. The host will issue a save state
* (via a bit in the command register) before it powers down the controller. The controller
* will use the scratch pad buffers (or its own memory if HCSPARAMS2:ScratchPadRestore == 0)
* to save its internal state before power shutdown (hibernation). If the SPR bit is set,
* the controller is to use the scratch pad buffers and the host is to make sure these
* buffers are maintained during hibernation. (The host may use paging to page to disk
* before hibernation as long as the memory is updated from the paged to disk before
* the restore is performed.)
*
* Since a Bochs power hibernation does not touch our current state, there would be no
* need to save the controller state. However, the host may (just might) check to see
* if the scratch pad buffers have been modified (due to paging and/or protection), so
* we need to write something to these buffers.
*
* We will simply save 'BX_XHCI_THIS hub' to the scratch pad buffers. However, since we
* use bx_bool's for bits, which are 32-bits for every single bit of the controller,
* 'hub' is way too big to fit in the scratch pad buffers. Therefore, we will only
* save up to PAGESIZE * BUFFER_COUNT.
*
* We will also use a simple crc check and store this as the last dword in each buffer.
* This way, when we restore, we can check to see if we restored the correct data and
* set the restore error bit accordingly.
*
* Please note that we must not assume that the scratch pad buffers will be in
* consecutive order, one right after the other, in memory.
*
* Callee checks to be sure the Halted bit is set first.
* We return 0 == no error, 1 = error
*/
bool bx_usb_xhci_c::save_hc_state(void)
{
int i, j;
Bit64u addr;
Bit32u *ptr = (Bit32u *) &BX_XHCI_THIS hub;
Bit64u temp[2 | MAX_SCRATCH_PADS]; // at least the larger of 2 or MAX_SCRATCH_PADS
Bit32u crc;
// do we use the scratch pad buffers to save the controller state?
#if (SCATCH_PAD_RESTORE == 1)
// get pointer to scratch pad buffers from DCBAAP
DEV_MEM_READ_PHYSICAL((bx_phy_address) BX_XHCI_THIS hub.op_regs.HcDCBAAP.dcbaap, 8, (Bit8u*)&addr);
// get MAX_SCRATCH_PADS worth of pointers
for (i=0; i<MAX_SCRATCH_PADS; i++) {
DEV_MEM_READ_PHYSICAL((bx_phy_address) (addr + i * 8), 8, (Bit8u*)&temp[i]);
}
for (i=0; i<MAX_SCRATCH_PADS; i++) {
crc = 0;
for (j=0; j<((XHCI_PAGE_SIZE << 12) >> 2) - 1; j++)
crc += ptr[j];
addr = temp[i];
DEV_MEM_WRITE_PHYSICAL_DMA((bx_phy_address) addr, (XHCI_PAGE_SIZE << 12) - sizeof(Bit32u), (Bit8u*)ptr);
DEV_MEM_WRITE_PHYSICAL((bx_phy_address) (addr + ((XHCI_PAGE_SIZE << 12) - sizeof(Bit32u))), sizeof(Bit32u), (Bit8u *) &crc);
ptr += (((XHCI_PAGE_SIZE << 12) >> 2) - 1);
}
#else
// Use controller's internal memory to save the state
// Since saving to the internal memory will have no effect on the host,
// since we don't destroy BX_XHCI_THIS hub on hibernation, we don't need
// to do anything here.
#endif
// return no error
return 0;
}
/*
* See comments at save_hc_state above
*/
bool bx_usb_xhci_c::restore_hc_state(void)
{
int i, j;
Bit64u addr;
Bit64u temp[2 | MAX_SCRATCH_PADS]; // at least the larger of 2 or MAX_SCRATCH_PADS
Bit32u temp_buffer[(XHCI_PAGE_SIZE << 12) >> 2];
Bit32u crc;
// if we are to use the scratch pad buffers to restore the controller state
#if (SCATCH_PAD_RESTORE == 1)
// get pointer to scratch pad buffers from DCBAAP
DEV_MEM_READ_PHYSICAL((bx_phy_address) BX_XHCI_THIS hub.op_regs.HcDCBAAP.dcbaap, 8, (Bit8u*)&addr);
// get MAX_SCRATCH_PADS worth of pointers
for (i=0; i<MAX_SCRATCH_PADS; i++) {
DEV_MEM_READ_PHYSICAL_DMA((bx_phy_address) (addr + i * 8), 8, (Bit8u*)&temp[i]);
}
// we read it in to a temp buffer just to check the crc.
for (i=0; i<MAX_SCRATCH_PADS; i++) {
addr = temp[i];
DEV_MEM_READ_PHYSICAL_DMA((bx_phy_address) addr, (XHCI_PAGE_SIZE << 12), (Bit8u*)temp_buffer);
crc = 0;
for (j=0; j<((XHCI_PAGE_SIZE << 12) >> 2) - 1; j++)
crc += temp_buffer[j];
if (crc != temp_buffer[((XHCI_PAGE_SIZE << 12) >> 2) - 1])
return 1; // error
}
#else
// Use controller's internal memory to save the state
// Since saving to the internal memory will have no effect on the host,
// since we don't destroy BX_XHCI_THIS hub on hibernation, we don't need
// to do anything here.
#endif
// return no error
return 0;
}
void bx_usb_xhci_c::register_state(void)
{
unsigned i, j;
char tmpname[16];
bx_list_c *hub, *port, *reg, *reg_grp, *reg_grp1;
bx_list_c *list1, *list2, *item, *entries, *entry;
bx_list_c *list = new bx_list_c(SIM->get_bochs_root(), "usb_xhci", "USB xHCI State");
hub = new bx_list_c(list, "hub");
reg_grp = new bx_list_c(hub, "op_regs");
reg = new bx_list_c(reg_grp, "HcCommand");
#if ((VERSION_MAJOR == 1) && (VERSION_MINOR >= 0x10))
BXRS_PARAM_BOOL(reg, cme, BX_XHCI_THIS hub.op_regs.HcCommand.cme);
BXRS_PARAM_BOOL(reg, spe, BX_XHCI_THIS hub.op_regs.HcCommand.spe);
#endif
BXRS_PARAM_BOOL(reg, eu3s, BX_XHCI_THIS hub.op_regs.HcCommand.eu3s);
BXRS_PARAM_BOOL(reg, ewe, BX_XHCI_THIS hub.op_regs.HcCommand.ewe);
BXRS_PARAM_BOOL(reg, crs, BX_XHCI_THIS hub.op_regs.HcCommand.crs);
BXRS_PARAM_BOOL(reg, css, BX_XHCI_THIS hub.op_regs.HcCommand.css);
BXRS_PARAM_BOOL(reg, lhcrst, BX_XHCI_THIS hub.op_regs.HcCommand.lhcrst);
BXRS_PARAM_BOOL(reg, hsee, BX_XHCI_THIS hub.op_regs.HcCommand.hsee);
BXRS_PARAM_BOOL(reg, inte, BX_XHCI_THIS hub.op_regs.HcCommand.inte);
BXRS_PARAM_BOOL(reg, hcrst, BX_XHCI_THIS hub.op_regs.HcCommand.hcrst);
BXRS_PARAM_BOOL(reg, rs, BX_XHCI_THIS hub.op_regs.HcCommand.rs);
reg = new bx_list_c(reg_grp, "HcStatus");
BXRS_PARAM_BOOL(reg, hce, BX_XHCI_THIS hub.op_regs.HcStatus.hce);
BXRS_PARAM_BOOL(reg, cnr, BX_XHCI_THIS hub.op_regs.HcStatus.cnr);
BXRS_PARAM_BOOL(reg, sre, BX_XHCI_THIS hub.op_regs.HcStatus.sre);
BXRS_PARAM_BOOL(reg, rss, BX_XHCI_THIS hub.op_regs.HcStatus.rss);
BXRS_PARAM_BOOL(reg, sss, BX_XHCI_THIS hub.op_regs.HcStatus.sss);
BXRS_PARAM_BOOL(reg, pcd, BX_XHCI_THIS hub.op_regs.HcStatus.pcd);
BXRS_PARAM_BOOL(reg, eint, BX_XHCI_THIS hub.op_regs.HcStatus.eint);
BXRS_PARAM_BOOL(reg, hse, BX_XHCI_THIS hub.op_regs.HcStatus.hse);
BXRS_PARAM_BOOL(reg, hch, BX_XHCI_THIS hub.op_regs.HcStatus.hch);
BXRS_HEX_PARAM_FIELD(reg_grp, HcPageSize, BX_XHCI_THIS hub.op_regs.HcPageSize.pagesize);
reg = new bx_list_c(reg_grp, "HcNotification");
BXRS_PARAM_BOOL(reg, n15, BX_XHCI_THIS hub.op_regs.HcNotification.n15);
BXRS_PARAM_BOOL(reg, n14, BX_XHCI_THIS hub.op_regs.HcNotification.n14);
BXRS_PARAM_BOOL(reg, n13, BX_XHCI_THIS hub.op_regs.HcNotification.n13);
BXRS_PARAM_BOOL(reg, n12, BX_XHCI_THIS hub.op_regs.HcNotification.n12);
BXRS_PARAM_BOOL(reg, n11, BX_XHCI_THIS hub.op_regs.HcNotification.n11);
BXRS_PARAM_BOOL(reg, n10, BX_XHCI_THIS hub.op_regs.HcNotification.n10);
BXRS_PARAM_BOOL(reg, n9, BX_XHCI_THIS hub.op_regs.HcNotification.n9);
BXRS_PARAM_BOOL(reg, n8, BX_XHCI_THIS hub.op_regs.HcNotification.n8);
BXRS_PARAM_BOOL(reg, n7, BX_XHCI_THIS hub.op_regs.HcNotification.n7);
BXRS_PARAM_BOOL(reg, n6, BX_XHCI_THIS hub.op_regs.HcNotification.n6);
BXRS_PARAM_BOOL(reg, n5, BX_XHCI_THIS hub.op_regs.HcNotification.n5);
BXRS_PARAM_BOOL(reg, n4, BX_XHCI_THIS hub.op_regs.HcNotification.n4);
BXRS_PARAM_BOOL(reg, n3, BX_XHCI_THIS hub.op_regs.HcNotification.n3);
BXRS_PARAM_BOOL(reg, n2, BX_XHCI_THIS hub.op_regs.HcNotification.n2);
BXRS_PARAM_BOOL(reg, n1, BX_XHCI_THIS hub.op_regs.HcNotification.n1);
BXRS_PARAM_BOOL(reg, n0, BX_XHCI_THIS hub.op_regs.HcNotification.n0);
reg = new bx_list_c(reg_grp, "HcCrcr");
BXRS_HEX_PARAM_FIELD(reg, crc, BX_XHCI_THIS hub.op_regs.HcCrcr.crc);
BXRS_PARAM_BOOL(reg, crr, BX_XHCI_THIS hub.op_regs.HcCrcr.crr);
BXRS_PARAM_BOOL(reg, ca, BX_XHCI_THIS hub.op_regs.HcCrcr.ca);
BXRS_PARAM_BOOL(reg, cs, BX_XHCI_THIS hub.op_regs.HcCrcr.cs);
BXRS_PARAM_BOOL(reg, rcs, BX_XHCI_THIS hub.op_regs.HcCrcr.rcs);
BXRS_HEX_PARAM_FIELD(reg_grp, HcDCBAAP, BX_XHCI_THIS hub.op_regs.HcDCBAAP.dcbaap);
BXRS_HEX_PARAM_FIELD(reg_grp, HcConfig_MaxSlotsEn, BX_XHCI_THIS hub.op_regs.HcConfig.MaxSlotsEn);
#if ((VERSION_MAJOR == 1) && (VERSION_MINOR >= 0x10))
BXRS_HEX_PARAM_FIELD(reg_grp, HcConfig_u3e, BX_XHCI_THIS hub.op_regs.HcConfig.u3e);
BXRS_HEX_PARAM_FIELD(reg_grp, HcConfig_cie, BX_XHCI_THIS hub.op_regs.HcConfig.cie);
#endif
for (i = 0; i < USB_XHCI_PORTS; i++) {
sprintf(tmpname, "port%d", i+1);
port = new bx_list_c(hub, tmpname);
BXRS_PARAM_BOOL(port, has_been_reset, BX_XHCI_THIS hub.usb_port[i].has_been_reset);
reg = new bx_list_c(port, "portsc");
BXRS_PARAM_BOOL(reg, wpr, BX_XHCI_THIS hub.usb_port[i].portsc.wpr);
BXRS_PARAM_BOOL(reg, dr, BX_XHCI_THIS hub.usb_port[i].portsc.dr);
BXRS_PARAM_BOOL(reg, woe, BX_XHCI_THIS hub.usb_port[i].portsc.woe);
BXRS_PARAM_BOOL(reg, wde, BX_XHCI_THIS hub.usb_port[i].portsc.wde);
BXRS_PARAM_BOOL(reg, wce, BX_XHCI_THIS hub.usb_port[i].portsc.wce);
BXRS_PARAM_BOOL(reg, cas, BX_XHCI_THIS hub.usb_port[i].portsc.cas);
BXRS_PARAM_BOOL(reg, cec, BX_XHCI_THIS hub.usb_port[i].portsc.cec);
BXRS_PARAM_BOOL(reg, plc, BX_XHCI_THIS hub.usb_port[i].portsc.plc);
BXRS_PARAM_BOOL(reg, prc, BX_XHCI_THIS hub.usb_port[i].portsc.prc);
BXRS_PARAM_BOOL(reg, occ, BX_XHCI_THIS hub.usb_port[i].portsc.occ);
BXRS_PARAM_BOOL(reg, wrc, BX_XHCI_THIS hub.usb_port[i].portsc.wrc);
BXRS_PARAM_BOOL(reg, pec, BX_XHCI_THIS hub.usb_port[i].portsc.pec);
BXRS_PARAM_BOOL(reg, csc, BX_XHCI_THIS hub.usb_port[i].portsc.csc);
BXRS_PARAM_BOOL(reg, lws, BX_XHCI_THIS hub.usb_port[i].portsc.lws);
BXRS_HEX_PARAM_FIELD(reg, pic, BX_XHCI_THIS hub.usb_port[i].portsc.pic);
BXRS_DEC_PARAM_FIELD(reg, speed, BX_XHCI_THIS hub.usb_port[i].portsc.speed);
BXRS_PARAM_BOOL(reg, pp, BX_XHCI_THIS hub.usb_port[i].portsc.pp);
BXRS_HEX_PARAM_FIELD(reg, pls, BX_XHCI_THIS hub.usb_port[i].portsc.pls);
BXRS_PARAM_BOOL(reg, pr, BX_XHCI_THIS hub.usb_port[i].portsc.pr);
BXRS_PARAM_BOOL(reg, oca, BX_XHCI_THIS hub.usb_port[i].portsc.oca);
BXRS_PARAM_BOOL(reg, ped, BX_XHCI_THIS hub.usb_port[i].portsc.ped);
BXRS_PARAM_BOOL(reg, ccs, BX_XHCI_THIS hub.usb_port[i].portsc.ccs);
reg = new bx_list_c(port, "portpmsc");
if (BX_XHCI_THIS hub.usb_port[i].is_usb3) {
BXRS_PARAM_BOOL(reg, fla, BX_XHCI_THIS hub.usb_port[i].usb3.portpmsc.fla);
BXRS_HEX_PARAM_FIELD(reg, u2timeout, BX_XHCI_THIS hub.usb_port[i].usb3.portpmsc.u2timeout);
BXRS_HEX_PARAM_FIELD(reg, u1timeout, BX_XHCI_THIS hub.usb_port[i].usb3.portpmsc.u1timeout);
BXRS_HEX_PARAM_FIELD(port, portli_lec, BX_XHCI_THIS hub.usb_port[i].usb3.portli.lec);
} else {
BXRS_HEX_PARAM_FIELD(reg, tmode, BX_XHCI_THIS hub.usb_port[i].usb2.portpmsc.tmode);
BXRS_PARAM_BOOL(reg, hle, BX_XHCI_THIS hub.usb_port[i].usb2.portpmsc.hle);
BXRS_HEX_PARAM_FIELD(reg, l1dslot, BX_XHCI_THIS hub.usb_port[i].usb2.portpmsc.l1dslot);
BXRS_HEX_PARAM_FIELD(reg, hird, BX_XHCI_THIS hub.usb_port[i].usb2.portpmsc.hird);
BXRS_PARAM_BOOL(reg, rwe, BX_XHCI_THIS hub.usb_port[i].usb2.portpmsc.rwe);
BXRS_HEX_PARAM_FIELD(reg, l1s, BX_XHCI_THIS hub.usb_port[i].usb2.portpmsc.l1s);
}
reg = new bx_list_c(port, "porthlpmc");
BXRS_HEX_PARAM_FIELD(reg, hirdm, BX_XHCI_THIS hub.usb_port[i].porthlpmc.hirdm);
BXRS_HEX_PARAM_FIELD(reg, l1timeout, BX_XHCI_THIS hub.usb_port[i].porthlpmc.l1timeout);
BXRS_HEX_PARAM_FIELD(reg, hirdd, BX_XHCI_THIS hub.usb_port[i].porthlpmc.hirdd);
// empty list for USB device state
new bx_list_c(port, "device");
}
new bx_shadow_data_c(hub, "extended_caps", BX_XHCI_THIS hub.extended_caps, EXT_CAPS_SIZE, 1);
reg_grp = new bx_list_c(hub, "runtime_regs");
BXRS_HEX_PARAM_FIELD(reg_grp, mfindex, BX_XHCI_THIS hub.runtime_regs.mfindex.index);
for (i = 0; i < INTERRUPTERS; i++) {
sprintf(tmpname, "interrupter%d", i+1);
reg_grp1 = new bx_list_c(reg_grp, tmpname);
reg = new bx_list_c(reg_grp1, "iman");
BXRS_PARAM_BOOL(reg, ie, BX_XHCI_THIS hub.runtime_regs.interrupter[i].iman.ie);
BXRS_PARAM_BOOL(reg, ip, BX_XHCI_THIS hub.runtime_regs.interrupter[i].iman.ip);
reg = new bx_list_c(reg_grp1, "imod");
BXRS_HEX_PARAM_FIELD(reg, imodc, BX_XHCI_THIS hub.runtime_regs.interrupter[i].imod.imodc);
BXRS_HEX_PARAM_FIELD(reg, imodi, BX_XHCI_THIS hub.runtime_regs.interrupter[i].imod.imodi);
BXRS_HEX_PARAM_FIELD(reg_grp1, erstabsize, BX_XHCI_THIS hub.runtime_regs.interrupter[i].erstsz.erstabsize);
BXRS_HEX_PARAM_FIELD(reg_grp1, erstabadd, BX_XHCI_THIS hub.runtime_regs.interrupter[i].erstba.erstabadd);
reg = new bx_list_c(reg_grp1, "erdp");
BXRS_HEX_PARAM_FIELD(reg, eventadd, BX_XHCI_THIS hub.runtime_regs.interrupter[i].erdp.eventadd);
BXRS_PARAM_BOOL(reg, ehb, BX_XHCI_THIS hub.runtime_regs.interrupter[i].erdp.ehb);
BXRS_HEX_PARAM_FIELD(reg, desi, BX_XHCI_THIS hub.runtime_regs.interrupter[i].erdp.desi);
}
list1 = new bx_list_c(hub, "slots");
for (i = 1; i < MAX_SLOTS; i++) {
sprintf(tmpname, "slot%d", i);
item = new bx_list_c(list1, tmpname);
BXRS_PARAM_BOOL(item, enabled, BX_XHCI_THIS hub.slots[i].enabled);
BXRS_PARAM_BOOL(item, sent_address, BX_XHCI_THIS hub.slots[i].sent_address);
list2 = new bx_list_c(item, "slot_context");
BXRS_DEC_PARAM_FIELD(list2, entries, BX_XHCI_THIS hub.slots[i].slot_context.entries);
BXRS_PARAM_BOOL(list2, hub, BX_XHCI_THIS hub.slots[i].slot_context.hub);
BXRS_PARAM_BOOL(list2, mtt, BX_XHCI_THIS hub.slots[i].slot_context.mtt);
BXRS_DEC_PARAM_FIELD(list2, speed, BX_XHCI_THIS hub.slots[i].slot_context.speed);
BXRS_DEC_PARAM_FIELD(list2, route_string, BX_XHCI_THIS hub.slots[i].slot_context.route_string);
BXRS_DEC_PARAM_FIELD(list2, num_ports, BX_XHCI_THIS hub.slots[i].slot_context.num_ports);
BXRS_DEC_PARAM_FIELD(list2, rh_port_num, BX_XHCI_THIS hub.slots[i].slot_context.rh_port_num);
BXRS_DEC_PARAM_FIELD(list2, max_exit_latency, BX_XHCI_THIS hub.slots[i].slot_context.max_exit_latency);
BXRS_DEC_PARAM_FIELD(list2, int_target, BX_XHCI_THIS hub.slots[i].slot_context.int_target);
BXRS_DEC_PARAM_FIELD(list2, ttt, BX_XHCI_THIS hub.slots[i].slot_context.ttt);
BXRS_DEC_PARAM_FIELD(list2, tt_port_num, BX_XHCI_THIS hub.slots[i].slot_context.tt_port_num);
BXRS_DEC_PARAM_FIELD(list2, tt_hub_slot_id, BX_XHCI_THIS hub.slots[i].slot_context.tt_hub_slot_id);
BXRS_DEC_PARAM_FIELD(list2, slot_state, BX_XHCI_THIS hub.slots[i].slot_context.slot_state);
BXRS_DEC_PARAM_FIELD(list2, device_address, BX_XHCI_THIS hub.slots[i].slot_context.device_address);
entries = new bx_list_c(item, "ep_context");
for (j = 0; j < 32; j++) {
sprintf(tmpname, "%d", j);
entry = new bx_list_c(entries, tmpname);
list2 = new bx_list_c(entry, "ep_context");
BXRS_DEC_PARAM_FIELD(list2, interval, BX_XHCI_THIS hub.slots[i].ep_context[j].ep_context.interval);
BXRS_PARAM_BOOL(list2, lsa, BX_XHCI_THIS hub.slots[i].ep_context[j].ep_context.lsa);
BXRS_DEC_PARAM_FIELD(list2, max_pstreams, BX_XHCI_THIS hub.slots[i].ep_context[j].ep_context.max_pstreams);
BXRS_DEC_PARAM_FIELD(list2, mult, BX_XHCI_THIS hub.slots[i].ep_context[j].ep_context.mult);
BXRS_DEC_PARAM_FIELD(list2, ep_state, BX_XHCI_THIS hub.slots[i].ep_context[j].ep_context.ep_state);
BXRS_DEC_PARAM_FIELD(list2, max_packet_size, BX_XHCI_THIS hub.slots[i].ep_context[j].ep_context.max_packet_size);
BXRS_DEC_PARAM_FIELD(list2, max_burst_size, BX_XHCI_THIS hub.slots[i].ep_context[j].ep_context.max_burst_size);
BXRS_PARAM_BOOL(list2, hid, BX_XHCI_THIS hub.slots[i].ep_context[j].ep_context.hid);
BXRS_DEC_PARAM_FIELD(list2, ep_type, BX_XHCI_THIS hub.slots[i].ep_context[j].ep_context.ep_type);
BXRS_DEC_PARAM_FIELD(list2, cerr, BX_XHCI_THIS hub.slots[i].ep_context[j].ep_context.cerr);
BXRS_HEX_PARAM_FIELD(list2, tr_dequeue_pointer, BX_XHCI_THIS hub.slots[i].ep_context[j].ep_context.tr_dequeue_pointer);
BXRS_PARAM_BOOL(list2, dcs, BX_XHCI_THIS hub.slots[i].ep_context[j].ep_context.dcs);
BXRS_DEC_PARAM_FIELD(list2, max_esit_payload, BX_XHCI_THIS hub.slots[i].ep_context[j].ep_context.max_esit_payload);
BXRS_DEC_PARAM_FIELD(list2, average_trb_len, BX_XHCI_THIS hub.slots[i].ep_context[j].ep_context.average_trb_len);
BXRS_HEX_PARAM_FIELD(entry, edtla, BX_XHCI_THIS hub.slots[i].ep_context[j].edtla);
BXRS_HEX_PARAM_FIELD(entry, enqueue_pointer, BX_XHCI_THIS hub.slots[i].ep_context[j].enqueue_pointer);
BXRS_PARAM_BOOL(entry, rcs, BX_XHCI_THIS hub.slots[i].ep_context[j].rcs);
BXRS_PARAM_BOOL(entry, retry, BX_XHCI_THIS hub.slots[i].ep_context[j].retry);
BXRS_DEC_PARAM_FIELD(entry, retry_counter, BX_XHCI_THIS hub.slots[i].ep_context[j].retry_counter);
}
}
list1 = new bx_list_c(hub, "ring_members");
list2 = new bx_list_c(list1, "command_ring");
BXRS_HEX_PARAM_FIELD(list2, dq_pointer, BX_XHCI_THIS hub.ring_members.command_ring.dq_pointer);
BXRS_PARAM_BOOL(list2, rcs, BX_XHCI_THIS hub.ring_members.command_ring.rcs);
list2 = new bx_list_c(list1, "event_rings");
for (i = 0; i < INTERRUPTERS; i++) {
sprintf(tmpname, "ring%d", i);
item = new bx_list_c(list2, tmpname);
BXRS_PARAM_BOOL(item, rcs, BX_XHCI_THIS hub.ring_members.event_rings[i].rcs);
BXRS_HEX_PARAM_FIELD(item, trb_count, BX_XHCI_THIS hub.ring_members.event_rings[i].trb_count);
BXRS_HEX_PARAM_FIELD(item, count, BX_XHCI_THIS hub.ring_members.event_rings[i].count);
BXRS_HEX_PARAM_FIELD(item, cur_trb, BX_XHCI_THIS hub.ring_members.event_rings[i].cur_trb);
entries = new bx_list_c(item, "entries");
for (j = 0; j < (1<<MAX_SEG_TBL_SZ_EXP); j++) {
sprintf(tmpname, "entry%d", j);
entry = new bx_list_c(entries, tmpname);
BXRS_HEX_PARAM_FIELD(entry, addr, BX_XHCI_THIS hub.ring_members.event_rings[i].entrys[j].addr);
BXRS_HEX_PARAM_FIELD(entry, size, BX_XHCI_THIS hub.ring_members.event_rings[i].entrys[j].size);
}
}
register_pci_state(hub);
}
void bx_usb_xhci_c::after_restore_state(void)
{
bx_pci_device_c::after_restore_pci_state(NULL);
for (int j=0; j<USB_XHCI_PORTS; j++) {
if (BX_XHCI_THIS hub.usb_port[j].device != NULL) {
BX_XHCI_THIS hub.usb_port[j].device->after_restore_state();
}
}
}
void bx_usb_xhci_c::init_device(Bit8u port, bx_list_c *portconf)
{
int type;
char pname[BX_PATHNAME_LEN];
const char *devname = NULL;
devname = ((bx_param_string_c*)portconf->get_by_name("device"))->getptr();
if (devname == NULL) return;
if (!strlen(devname) || !strcmp(devname, "none")) return;
if (BX_XHCI_THIS hub.usb_port[port].device != NULL) {
BX_ERROR(("init_device(): port%d already in use", port+1));
return;
}
sprintf(pname, "usb_xhci.hub.port%d.device", port+1);
bx_list_c *sr_list = (bx_list_c*)SIM->get_param(pname, SIM->get_bochs_root());
type = DEV_usb_init_device(portconf, BX_XHCI_THIS_PTR, &BX_XHCI_THIS hub.usb_port[port].device);
if (BX_XHCI_THIS hub.usb_port[port].device != NULL) {
usb_set_connect_status(port, type, 1);
BX_XHCI_THIS hub.usb_port[port].device->register_state(sr_list);
}
}
void bx_usb_xhci_c::remove_device(Bit8u port)
{
if (BX_XHCI_THIS hub.usb_port[port].device != NULL) {
delete BX_XHCI_THIS hub.usb_port[port].device;
BX_XHCI_THIS hub.usb_port[port].device = NULL;
}
}
void bx_usb_xhci_c::update_irq(unsigned interrupter)
{
bool level = 0;
if ((BX_XHCI_THIS hub.op_regs.HcCommand.inte) &&
(BX_XHCI_THIS hub.runtime_regs.interrupter[interrupter].iman.ie)) {
level = 1;
BX_DEBUG(("Interrupt Fired."));
}
DEV_pci_set_irq(BX_XHCI_THIS devfunc, BX_XHCI_THIS pci_conf[0x3d], level);
}
bool bx_usb_xhci_c::read_handler(bx_phy_address addr, unsigned len, void *data, void *param)
{
Bit32u val = 0, val_hi = 0;
int i, speed = 0;
const Bit32u offset = (Bit32u) (addr - BX_XHCI_THIS pci_bar[0].addr);
// Even though the controller allows reads other than 32-bits & on odd boundaries,
// we are going to ASSUME dword reads and writes unless specified below
// RO Capability Registers
if (offset < OPS_REGS_OFFSET) {
switch (offset) {
// Capability Registers
case 0x00: // CapLength / version
val = BX_XHCI_THIS hub.cap_regs.HcCapLength;
break;
case 0x01:
val = BX_XHCI_THIS hub.cap_regs.HcCapLength >> 8;
break;
case 0x02:
val = BX_XHCI_THIS hub.cap_regs.HcCapLength >> 16;
break;
case 0x04: // HCSPARAMS1
val = BX_XHCI_THIS hub.cap_regs.HcSParams1;
break;
case 0x08: // HCSPARAMS2
val = BX_XHCI_THIS hub.cap_regs.HcSParams2;
break;
case 0x0C: // HCSPARAMS3
val = BX_XHCI_THIS hub.cap_regs.HcSParams3;
break;
case 0x10: // HCCPARAMS1
val = BX_XHCI_THIS hub.cap_regs.HcCParams1;
break;
case 0x14: // DBOFF
val = BX_XHCI_THIS hub.cap_regs.DBOFF;
break;
case 0x18: // RTSOFF
val = BX_XHCI_THIS hub.cap_regs.RTSOFF;
break;
#if ((VERSION_MAJOR == 1) && (VERSION_MINOR >= 0x10))
case 0x1C: // HCCPARAMS2
val = BX_XHCI_THIS hub.cap_regs.HcCParams2;
break;
#else
case 0x1C: // reserved
val = 0;
break;
#endif
}
} else
// Operational Registers
if ((offset >= OPS_REGS_OFFSET) && (offset < (OPS_REGS_OFFSET + 0x40))) {
switch (offset - OPS_REGS_OFFSET) {
case 0x00: // Command
#if ((VERSION_MAJOR == 1) && (VERSION_MINOR >= 0x10))
val = (BX_XHCI_THIS hub.op_regs.HcCommand.RsvdP1 << 14)
| (BX_XHCI_THIS hub.op_regs.HcCommand.cme ? 1 << 13 : 0)
| (BX_XHCI_THIS hub.op_regs.HcCommand.spe ? 1 << 12 : 0)
#else
val = (BX_XHCI_THIS hub.op_regs.HcCommand.RsvdP1 << 12)
#endif
| (BX_XHCI_THIS hub.op_regs.HcCommand.eu3s ? 1 << 11 : 0)
| (BX_XHCI_THIS hub.op_regs.HcCommand.ewe ? 1 << 10 : 0)
| 0 // controller restore state always read as zero
| 0 // controller save state always read as zero
| (BX_XHCI_THIS hub.op_regs.HcCommand.lhcrst ? 1 << 7 : 0)
| (BX_XHCI_THIS hub.op_regs.HcCommand.RsvdP0 << 4)
| (BX_XHCI_THIS hub.op_regs.HcCommand.hsee ? 1 << 3 : 0)
| (BX_XHCI_THIS hub.op_regs.HcCommand.inte ? 1 << 2 : 0)
| (BX_XHCI_THIS hub.op_regs.HcCommand.hcrst ? 1 << 1 : 0)
| (BX_XHCI_THIS hub.op_regs.HcCommand.rs ? 1 << 0 : 0);
break;
case 0x04: // Status
val = (BX_XHCI_THIS hub.op_regs.HcStatus.hce ? 1 << 12 : 0)
| (BX_XHCI_THIS hub.op_regs.HcStatus.cnr ? 1 << 11 : 0)
| (BX_XHCI_THIS hub.op_regs.HcStatus.sre ? 1 << 10 : 0)
| (BX_XHCI_THIS hub.op_regs.HcStatus.rss ? 1 << 9 : 0)
| (BX_XHCI_THIS hub.op_regs.HcStatus.sss ? 1 << 8 : 0)
| (BX_XHCI_THIS hub.op_regs.HcStatus.pcd ? 1 << 4 : 0)
| (BX_XHCI_THIS hub.op_regs.HcStatus.eint ? 1 << 3 : 0)
| (BX_XHCI_THIS hub.op_regs.HcStatus.hse ? 1 << 2 : 0)
| (BX_XHCI_THIS hub.op_regs.HcStatus.hch ? 1 << 0 : 0);
break;
case 0x08: // Page Size
val = BX_XHCI_THIS hub.op_regs.HcPageSize.pagesize;
break;
case 0x0C: // Reserved and Zero'd
case 0x10: // Reserved and Zero'd
val = 0;
break;
case 0x14: // Device Notification Control Register
val = (BX_XHCI_THIS hub.op_regs.HcNotification.RsvdP << 16)
| (BX_XHCI_THIS hub.op_regs.HcNotification.n15 ? 1 << 15 : 0)
| (BX_XHCI_THIS hub.op_regs.HcNotification.n14 ? 1 << 14 : 0)
| (BX_XHCI_THIS hub.op_regs.HcNotification.n13 ? 1 << 13 : 0)
| (BX_XHCI_THIS hub.op_regs.HcNotification.n12 ? 1 << 12 : 0)
| (BX_XHCI_THIS hub.op_regs.HcNotification.n11 ? 1 << 11 : 0)
| (BX_XHCI_THIS hub.op_regs.HcNotification.n10 ? 1 << 10 : 0)
| (BX_XHCI_THIS hub.op_regs.HcNotification.n9 ? 1 << 9 : 0)
| (BX_XHCI_THIS hub.op_regs.HcNotification.n8 ? 1 << 8 : 0)
| (BX_XHCI_THIS hub.op_regs.HcNotification.n7 ? 1 << 7 : 0)
| (BX_XHCI_THIS hub.op_regs.HcNotification.n6 ? 1 << 6 : 0)
| (BX_XHCI_THIS hub.op_regs.HcNotification.n5 ? 1 << 5 : 0)
| (BX_XHCI_THIS hub.op_regs.HcNotification.n4 ? 1 << 4 : 0)
| (BX_XHCI_THIS hub.op_regs.HcNotification.n3 ? 1 << 3 : 0)
| (BX_XHCI_THIS hub.op_regs.HcNotification.n2 ? 1 << 2 : 0)
| (BX_XHCI_THIS hub.op_regs.HcNotification.n1 ? 1 << 1 : 0)
| (BX_XHCI_THIS hub.op_regs.HcNotification.n0 ? 1 << 0 : 0);
break;
case 0x18: // Command Ring Control Register (Lo) (most fields always returns zero when read)
val = (BX_XHCI_THIS hub.op_regs.HcCrcr.RsvdP << 4)
| (BX_XHCI_THIS hub.op_regs.HcCrcr.crr ? 1 << 3 : 0);
break;
case 0x1C: // Command Ring Control Register (Hi) (always returns zero when read)
val = 0;
break;
case 0x20: // Reserved and Zero'd
case 0x24: // Reserved and Zero'd
case 0x28: // Reserved and Zero'd
case 0x2C: // Reserved and Zero'd
val = 0;
break;
case 0x30: // DCBAAP (Lo)
val = (Bit32u) ((Bit32u) BX_XHCI_THIS hub.op_regs.HcDCBAAP.dcbaap & ~0x3F);
if (len == 8)
#if ADDR_CAP_64
val_hi = (Bit32u) (BX_XHCI_THIS hub.op_regs.HcDCBAAP.dcbaap >> 32);
#else
val_hi = 0;
#endif // ADDR_CAP_64
break;
case 0x34: // DCBAAP (Hi)
#if ADDR_CAP_64
val = (Bit32u) (BX_XHCI_THIS hub.op_regs.HcDCBAAP.dcbaap >> 32);
#else
val = 0;
#endif // ADDR_CAP_64
break;
case 0x38: // Config
#if ((VERSION_MAJOR == 1) && (VERSION_MINOR >= 0x10))
val = (BX_XHCI_THIS hub.op_regs.HcConfig.RsvdP << 10)
| (BX_XHCI_THIS hub.op_regs.HcConfig.cie << 8)
| (BX_XHCI_THIS hub.op_regs.HcConfig.u3e << 9)
#else
val = (BX_XHCI_THIS hub.op_regs.HcConfig.RsvdP << 8)
#endif
| (BX_XHCI_THIS hub.op_regs.HcConfig.MaxSlotsEn << 0);
break;
}
} else
// Register Port Sets
if ((offset >= PORT_SET_OFFSET) && (offset < (PORT_SET_OFFSET + (USB_XHCI_PORTS * 16)))) {
unsigned port = (((offset - PORT_SET_OFFSET) >> 4) & 0x3F); // calculate port number
if (BX_XHCI_THIS hub.usb_port[port].portsc.pp) {
// the speed field is only valid for USB3 before a port reset. If a reset has not
// taken place after the port is powered, the USB2 ports don't show a valid speed field.
if (BX_XHCI_THIS hub.usb_port[port].portsc.ccs &&
(BX_XHCI_THIS hub.usb_port[port].is_usb3 || BX_XHCI_THIS hub.usb_port[port].has_been_reset))
speed = BX_XHCI_THIS hub.usb_port[port].portsc.speed;
switch (offset & 0x0000000F) {
case 0x00:
val = 0 /* BX_XHCI_THIS hub.usb_port[port].portsc.wpr == 0 when read */
| (BX_XHCI_THIS hub.usb_port[port].portsc.dr ? 1 << 30 : 0)
| (BX_XHCI_THIS hub.usb_port[port].portsc.woe ? 1 << 27 : 0)
| (BX_XHCI_THIS hub.usb_port[port].portsc.wde ? 1 << 26 : 0)
| (BX_XHCI_THIS hub.usb_port[port].portsc.wce ? 1 << 25 : 0)
| (BX_XHCI_THIS hub.usb_port[port].portsc.cas ? 1 << 24 : 0)
| (BX_XHCI_THIS hub.usb_port[port].portsc.cec ? 1 << 23 : 0)
| (BX_XHCI_THIS hub.usb_port[port].portsc.plc ? 1 << 22 : 0)
| (BX_XHCI_THIS hub.usb_port[port].portsc.prc ? 1 << 21 : 0)
| (BX_XHCI_THIS hub.usb_port[port].portsc.occ ? 1 << 20 : 0)
| (BX_XHCI_THIS hub.usb_port[port].portsc.wrc ? 1 << 19 : 0)
| (BX_XHCI_THIS hub.usb_port[port].portsc.pec ? 1 << 18 : 0)
| (BX_XHCI_THIS hub.usb_port[port].portsc.csc ? 1 << 17 : 0)
| (BX_XHCI_THIS hub.usb_port[port].portsc.pic << 14)
| ( speed << 10)
| (BX_XHCI_THIS hub.usb_port[port].portsc.pp ? 1 << 9 : 0)
| (BX_XHCI_THIS hub.usb_port[port].portsc.pls << 5)
| (BX_XHCI_THIS hub.usb_port[port].portsc.pr ? 1 << 4 : 0)
| (BX_XHCI_THIS hub.usb_port[port].portsc.oca ? 1 << 3 : 0)
| (BX_XHCI_THIS hub.usb_port[port].portsc.ped ? 1 << 1 : 0)
| (BX_XHCI_THIS hub.usb_port[port].portsc.ccs ? 1 << 0 : 0);
break;
case 0x04:
if (BX_XHCI_THIS hub.usb_port[port].is_usb3) {
val = (BX_XHCI_THIS hub.usb_port[port].usb3.portpmsc.RsvdP << 17)
| (BX_XHCI_THIS hub.usb_port[port].usb3.portpmsc.fla ? 1 << 16 : 0)
| (BX_XHCI_THIS hub.usb_port[port].usb3.portpmsc.u2timeout << 8)
| (BX_XHCI_THIS hub.usb_port[port].usb3.portpmsc.u1timeout << 0);
} else {
val = (BX_XHCI_THIS hub.usb_port[port].usb2.portpmsc.tmode << 28)
| (BX_XHCI_THIS hub.usb_port[port].usb2.portpmsc.RsvdP << 17)
| (BX_XHCI_THIS hub.usb_port[port].usb2.portpmsc.hle ? 1 << 16 : 0)
| (BX_XHCI_THIS hub.usb_port[port].usb2.portpmsc.l1dslot << 8)
| (BX_XHCI_THIS hub.usb_port[port].usb2.portpmsc.hird << 4)
| (BX_XHCI_THIS hub.usb_port[port].usb2.portpmsc.rwe ? 1 << 3 : 0)
| (BX_XHCI_THIS hub.usb_port[port].usb2.portpmsc.l1s << 0);
}
break;
case 0x08:
if (BX_XHCI_THIS hub.usb_port[port].is_usb3) {
val = (BX_XHCI_THIS hub.usb_port[port].usb3.portli.RsvdP << 16)
| (BX_XHCI_THIS hub.usb_port[port].usb3.portli.lec << 0);
} else {
val = BX_XHCI_THIS hub.usb_port[port].usb2.portli.RsvdP;
}
break;
case 0x0C:
#if ((VERSION_MAJOR < 1) || ((VERSION_MAJOR == 1) && (VERSION_MINOR == 0)))
BX_ERROR(("Read from Reserved Register in Port Register Set %i", port));
#else
val = (BX_XHCI_THIS hub.usb_port[port].porthlpmc.RsvdP << 14)
| (BX_XHCI_THIS hub.usb_port[port].porthlpmc.hirdd << 10)
| (BX_XHCI_THIS hub.usb_port[port].porthlpmc.l1timeout << 2)
| (BX_XHCI_THIS hub.usb_port[port].porthlpmc.hirdm << 0);
#endif
break;
}
} else val = 0;
} else
// Extended Capabilities
if ((offset >= EXT_CAPS_OFFSET) && (offset < (EXT_CAPS_OFFSET + EXT_CAPS_SIZE))) {
unsigned caps_offset = (offset - EXT_CAPS_OFFSET);
switch (len) {
case 1:
val = BX_XHCI_THIS hub.extended_caps[caps_offset];
break;
case 2:
val = BX_XHCI_THIS hub.extended_caps[caps_offset] |
(BX_XHCI_THIS hub.extended_caps[caps_offset + 1] << 8);
break;
case 8:
val_hi = (BX_XHCI_THIS hub.extended_caps[caps_offset + 4] |
(BX_XHCI_THIS hub.extended_caps[caps_offset + 5] << 8) |
(BX_XHCI_THIS hub.extended_caps[caps_offset + 6] << 16) |
(BX_XHCI_THIS hub.extended_caps[caps_offset + 7] << 24));
case 4:
val = (BX_XHCI_THIS hub.extended_caps[caps_offset] |
(BX_XHCI_THIS hub.extended_caps[caps_offset + 1] << 8) |
(BX_XHCI_THIS hub.extended_caps[caps_offset + 2] << 16) |
(BX_XHCI_THIS hub.extended_caps[caps_offset + 3] << 24));
break;
}
} else
// Host Controller Runtime Registers
if ((offset >= RUNTIME_OFFSET) && (offset < (RUNTIME_OFFSET + 32 + (INTERRUPTERS * 32)))) {
if (offset == RUNTIME_OFFSET) {
val = (BX_XHCI_THIS hub.runtime_regs.mfindex.RsvdP << 14)
| (BX_XHCI_THIS hub.runtime_regs.mfindex.index << 0);
} else if (offset < (RUNTIME_OFFSET + 32)) {
val = 0;
} else {
unsigned rt_offset = (offset - RUNTIME_OFFSET - 32);
i = (rt_offset >> 5); // interrupter offset
switch (rt_offset & 0x1F) {
case 0x00:
val = (BX_XHCI_THIS hub.runtime_regs.interrupter[i].iman.RsvdP << 2)
| (BX_XHCI_THIS hub.runtime_regs.interrupter[i].iman.ie ? 1 << 1 : 0)
| (BX_XHCI_THIS hub.runtime_regs.interrupter[i].iman.ip ? 1 << 0 : 0);
break;
case 0x04:
val = (BX_XHCI_THIS hub.runtime_regs.interrupter[i].imod.imodc << 16)
| (BX_XHCI_THIS hub.runtime_regs.interrupter[i].imod.imodi << 0);
break;
case 0x08:
val = (BX_XHCI_THIS hub.runtime_regs.interrupter[i].erstsz.RsvdP << 16)
| (BX_XHCI_THIS hub.runtime_regs.interrupter[i].erstsz.erstabsize << 0);
break;
case 0x0C:
val = BX_XHCI_THIS hub.runtime_regs.interrupter[i].RsvdP;
break;
case 0x10:
val = ((Bit32u) BX_XHCI_THIS hub.runtime_regs.interrupter[i].erstba.erstabadd & ~0x3F)
| (BX_XHCI_THIS hub.runtime_regs.interrupter[i].erstba.RsvdP << 0);
if (len == 8)
#if ADDR_CAP_64
val_hi = (Bit32u) (BX_XHCI_THIS hub.runtime_regs.interrupter[i].erstba.erstabadd >> 32);
#else
val_hi = 0;
#endif // ADDR_CAP_64
break;
case 0x14:
#if ADDR_CAP_64
val = (Bit32u) (BX_XHCI_THIS hub.runtime_regs.interrupter[i].erstba.erstabadd >> 32);
#else
val = 0;
#endif // ADDR_CAP_64
break;
case 0x18:
val = (Bit32u) ((Bit32u) BX_XHCI_THIS hub.runtime_regs.interrupter[i].erdp.eventadd & ~0x0F)
| (BX_XHCI_THIS hub.runtime_regs.interrupter[i].erdp.ehb ? 1 << 3 : 0)
| (BX_XHCI_THIS hub.runtime_regs.interrupter[i].erdp.desi << 0);
if (len == 8)
#if ADDR_CAP_64
val_hi = (Bit32u) (BX_XHCI_THIS hub.runtime_regs.interrupter[i].erdp.eventadd >> 32);
#else
val_hi = 0;
#endif // ADDR_CAP_64
break;
case 0x1C:
#if ADDR_CAP_64
val = (Bit32u) (BX_XHCI_THIS hub.runtime_regs.interrupter[i].erdp.eventadd >> 32);
#else
val = 0;
#endif // ADDR_CAP_64
break;
}
}
} else
// Doorbell Registers (return zero when read)
if ((offset >= DOORBELL_OFFSET) && (offset < (DOORBELL_OFFSET + 4 + (INTERRUPTERS * 4)))) {
val = 0;
} else {
#if BX_PHY_ADDRESS_LONG
BX_ERROR(("register read from unknown offset 0x%08X: 0x%08X%08X (len=%i)", offset, (Bit32u) val_hi, (Bit32u) val, len));
#else
BX_DEBUG(("register read from unknown offset 0x%08X: 0x%08X (len=%i)", offset, (Bit32u) val, len));
if (len > 4)
BX_DEBUG(("Ben: In 32-bit mode, len > 4! (len=%i)", len));
#endif
}
switch (len) {
case 1:
val &= 0xFF;
*((Bit8u *) data) = (Bit8u) val;
break;
case 2:
val &= 0xFFFF;
*((Bit16u *) data) = (Bit16u) val;
break;
case 8:
*((Bit32u *) ((Bit8u *) data + 4)) = val_hi;
case 4:
*((Bit32u *) data) = val;
break;
}
// don't populate the log file if reading from interrupter's IMAN and only INT_ENABLE is set.
// (This only works with the first interrupter)
if ((offset != 0x620) || (val != 0x02)) {
#if BX_PHY_ADDRESS_LONG
BX_DEBUG(("register read from offset 0x%04X: 0x%08X%08X (len=%i)", offset, (Bit32u) val_hi, (Bit32u) val, len));
#else
BX_DEBUG(("register read from offset 0x%04X: 0x%08X (len=%i)", offset, (Bit32u) val, len));
if (len > 4)
BX_DEBUG(("Ben: In 32-bit mode, len > 4! (len=%i)", len));
#endif
}
return 1;
}
bool bx_usb_xhci_c::write_handler(bx_phy_address addr, unsigned len, void *data, void *param)
{
Bit32u value = *((Bit32u *) data);
Bit32u value_hi = *((Bit32u *) ((Bit8u *) data + 4));
const Bit32u offset = (Bit32u) (addr - BX_XHCI_THIS pci_bar[0].addr);
Bit32u temp;
int i;
// modify val and val_hi per len of data to write
switch (len) {
case 1:
value &= 0xFF;
case 2:
value &= 0xFFFF;
case 4:
value_hi = 0;
break;
}
#if BX_PHY_ADDRESS_LONG
BX_DEBUG(("register write to offset 0x%04X: 0x%08X%08X (len=%i)", offset, value_hi, value, len));
#else
BX_DEBUG(("register write to offset 0x%04X: 0x%08X (len=%i)", offset, value, len));
if (len > 4)
BX_DEBUG(("Ben: In 32-bit mode, len > 4! (len=%i)", len));
#endif
// Even though the controller allows reads other than 32-bits & on odd boundaries,
// we are going to ASSUME dword reads and writes unless specified below
// RO Capability Registers
if (offset < OPS_REGS_OFFSET) {
switch (offset) {
// Capability Registers
case 0x00: // CapLength / version
case 0x04: // HCSPARAMS1
case 0x08: // HCSPARAMS2
case 0x0C: // HCSPARAMS3
case 0x10: // HCCPARAMS1
case 0x14: // DBOFF
case 0x18: // TRSOFF
case 0x1C: // reserved/HCCPARAMS2
BX_ERROR(("Write to Read Only Host Capability Register (0x%08X)", offset));
break;
}
} else
// Operational Registers
if ((offset >= OPS_REGS_OFFSET) && (offset < (OPS_REGS_OFFSET + 0x40))) {
switch (offset - OPS_REGS_OFFSET) {
case 0x00: // Command
temp = BX_XHCI_THIS hub.op_regs.HcCommand.RsvdP1;
#if ((VERSION_MAJOR == 1) && (VERSION_MINOR >= 0x10))
BX_XHCI_THIS hub.op_regs.HcCommand.RsvdP1 = (value >> 14);
if (temp != BX_XHCI_THIS hub.op_regs.HcCommand.RsvdP1)
BX_ERROR(("bits 31:14 in command register were not preserved"));
BX_XHCI_THIS hub.op_regs.HcCommand.cme = (value & (1 << 13)) ? 1 : 0;
BX_XHCI_THIS hub.op_regs.HcCommand.spe = (value & (1 << 12)) ? 1 : 0;
#else
BX_XHCI_THIS hub.op_regs.HcCommand.RsvdP1 = (value >> 12);
if (temp != BX_XHCI_THIS hub.op_regs.HcCommand.RsvdP1)
BX_ERROR(("bits 31:12 in command register were not preserved"));
#endif
BX_XHCI_THIS hub.op_regs.HcCommand.eu3s = (value & (1 << 11)) ? 1 : 0;
BX_XHCI_THIS hub.op_regs.HcCommand.ewe = (value & (1 << 10)) ? 1 : 0;
if (value & (1 << 9)) {
if (BX_XHCI_THIS hub.op_regs.HcStatus.hch == 1) {
// show that we are restoring the state
// (redundant since emulated host can't see it set until we are done here)
BX_XHCI_THIS hub.op_regs.HcStatus.rss = 1;
BX_XHCI_THIS hub.op_regs.HcStatus.sre = BX_XHCI_THIS restore_hc_state();
// show that the restoring is done. (again, redundant, but we do it anyway)
BX_XHCI_THIS hub.op_regs.HcStatus.rss = 0;
} else
BX_ERROR(("Restore State when controller not in Halted state."));
}
if (value & (1 << 8)) {
if (BX_XHCI_THIS hub.op_regs.HcStatus.hch == 1) {
// show that we are saving the state
// (redundant since emulated host can't see it set until we are done here)
BX_XHCI_THIS hub.op_regs.HcStatus.sss = 1;
BX_XHCI_THIS hub.op_regs.HcStatus.sre = BX_XHCI_THIS save_hc_state();
// show that the saving is done. (again, redundant, but we do it anyway)
BX_XHCI_THIS hub.op_regs.HcStatus.sss = 0;
} else
BX_ERROR(("Save State when controller not in Halted state."));
}
#if LIGHT_HC_RESET
BX_XHCI_THIS hub.op_regs.HcCommand.lhcrst = (value & (1 << 7)) ? 1 : 0;
// TODO: Do light reset
#endif // LIGHT_HC_RESET
temp = BX_XHCI_THIS hub.op_regs.HcCommand.RsvdP0;
BX_XHCI_THIS hub.op_regs.HcCommand.RsvdP0 = (value & (7 << 4)) >> 4;
if (temp != BX_XHCI_THIS hub.op_regs.HcCommand.RsvdP0)
BX_ERROR(("bits 6:4 in Command Register were not preserved"));
BX_XHCI_THIS hub.op_regs.HcCommand.hsee = (value & (1 << 3)) ? 1 : 0;
BX_XHCI_THIS hub.op_regs.HcCommand.inte = (value & (1 << 2)) ? 1 : 0;
BX_XHCI_THIS hub.op_regs.HcCommand.hcrst = (value & (1 << 1)) ? 1 : 0;
if (BX_XHCI_THIS hub.op_regs.HcCommand.hcrst) {
BX_XHCI_THIS reset_hc();
BX_XHCI_THIS hub.op_regs.HcCommand.hcrst = 0;
// the controller will send a reset to all USB 3.0 ports,
// enabling the port if a device is attached.
for (int port=0; port<USB_XHCI_PORTS; port++) {
if (BX_XHCI_THIS hub.usb_port[port].is_usb3) {
reset_port_usb3(port, HOT_RESET);
}
}
}
// if run/stop bit cleared, stop command ring
BX_XHCI_THIS hub.op_regs.HcCommand.rs = (value & (1 << 0)) ? 1 : 0;
if (BX_XHCI_THIS hub.op_regs.HcCommand.rs == 0) {
BX_XHCI_THIS hub.op_regs.HcCrcr.crr = 0;
BX_XHCI_THIS hub.op_regs.HcStatus.hch = 1; // set the Halted Bit
} else
BX_XHCI_THIS hub.op_regs.HcStatus.hch = 0; // clear the Halted Bit
break;
case 0x04: // Status
if ((value & 0xFFFFE0E2) != 0)
BX_ERROR(("RsvdZ bits in HcStatus not written as zeros"));
if (value & ((1<<12) | (1<<11) | (1<<9) | (1<<8) | (1<<0)))
BX_ERROR(("Write to one or more Read-only bits in Status Register"));
BX_XHCI_THIS hub.op_regs.HcStatus.sre = (value & (1 << 10)) ? 0 : BX_XHCI_THIS hub.op_regs.HcStatus.sre;
BX_XHCI_THIS hub.op_regs.HcStatus.pcd = (value & (1 << 4)) ? 0 : BX_XHCI_THIS hub.op_regs.HcStatus.pcd;
BX_XHCI_THIS hub.op_regs.HcStatus.eint = (value & (1 << 3)) ? 0 : BX_XHCI_THIS hub.op_regs.HcStatus.eint;
BX_XHCI_THIS hub.op_regs.HcStatus.hse = (value & (1 << 2)) ? 0 : BX_XHCI_THIS hub.op_regs.HcStatus.hse;
//FIXME: should this line go where system software clears the IP bit, or here when it clears the status:eint bit?
if (value & (1 << 3)) // acknowledging the interrupt
DEV_pci_set_irq(BX_XHCI_THIS devfunc, BX_XHCI_THIS pci_conf[0x3d], 0);
break;
case 0x08: // Page Size
BX_ERROR(("Write to one or more Read-only bits in Page Size Register"));
break;
case 0x0C: // Reserved and Zero'd
case 0x10: // Reserved and Zero'd
if (value != 0)
BX_ERROR(("Write non-zero to RsvdZ Register (offset = 0x%08X value = 0x%08X)", offset, value));
break;
case 0x14: // Device Notification Control Register
temp = BX_XHCI_THIS hub.op_regs.HcNotification.RsvdP;
BX_XHCI_THIS hub.op_regs.HcNotification.RsvdP = value >> 16;
if (temp != BX_XHCI_THIS hub.op_regs.HcNotification.RsvdP)
BX_ERROR(("bits 31:16 in DNCTRL Register were not preserved"));
BX_XHCI_THIS hub.op_regs.HcNotification.n15 = (value & (1 << 15)) ? 1 : 0;
BX_XHCI_THIS hub.op_regs.HcNotification.n14 = (value & (1 << 14)) ? 1 : 0;
BX_XHCI_THIS hub.op_regs.HcNotification.n13 = (value & (1 << 13)) ? 1 : 0;
BX_XHCI_THIS hub.op_regs.HcNotification.n12 = (value & (1 << 12)) ? 1 : 0;
BX_XHCI_THIS hub.op_regs.HcNotification.n11 = (value & (1 << 11)) ? 1 : 0;
BX_XHCI_THIS hub.op_regs.HcNotification.n10 = (value & (1 << 10)) ? 1 : 0;
BX_XHCI_THIS hub.op_regs.HcNotification.n9 = (value & (1 << 9)) ? 1 : 0;
BX_XHCI_THIS hub.op_regs.HcNotification.n8 = (value & (1 << 8)) ? 1 : 0;
BX_XHCI_THIS hub.op_regs.HcNotification.n7 = (value & (1 << 7)) ? 1 : 0;
BX_XHCI_THIS hub.op_regs.HcNotification.n6 = (value & (1 << 6)) ? 1 : 0;
BX_XHCI_THIS hub.op_regs.HcNotification.n5 = (value & (1 << 5)) ? 1 : 0;
BX_XHCI_THIS hub.op_regs.HcNotification.n4 = (value & (1 << 4)) ? 1 : 0;
BX_XHCI_THIS hub.op_regs.HcNotification.n3 = (value & (1 << 3)) ? 1 : 0;
BX_XHCI_THIS hub.op_regs.HcNotification.n2 = (value & (1 << 2)) ? 1 : 0;
BX_XHCI_THIS hub.op_regs.HcNotification.n1 = (value & (1 << 1)) ? 1 : 0;
BX_XHCI_THIS hub.op_regs.HcNotification.n0 = (value & (1 << 0)) ? 1 : 0;
break;
case 0x18: // Command Ring Control Register (Lo)
if (value & (1<<3))
BX_ERROR(("Write to one or more Read-only bits in CRCR Register"));
temp = BX_XHCI_THIS hub.op_regs.HcCrcr.RsvdP;
BX_XHCI_THIS hub.op_regs.HcCrcr.RsvdP = (value & (2<<4)) >> 4;
if (temp != BX_XHCI_THIS hub.op_regs.HcCrcr.RsvdP)
BX_ERROR(("bits 5:4 in CRCR Register were not preserved"));
BX_XHCI_THIS hub.op_regs.HcCrcr.ca = (value & (1 << 2)) ? 1 : 0;
BX_XHCI_THIS hub.op_regs.HcCrcr.cs = (value & (1 << 1)) ? 1 : 0;
BX_XHCI_THIS hub.op_regs.HcCrcr.rcs = (value & (1 << 0)) ? 1 : 0;
// save the new ring pointer
#if ADDR_CAP_64
if (len == 8) {
BX_XHCI_THIS hub.op_regs.HcCrcr.crc = (Bit64u) (((Bit64u) value_hi << 32) | (value & ~0x3F));
} else {
BX_XHCI_THIS hub.op_regs.HcCrcr.crc &= (Bit64u) ~0xFFFFFFFF;
BX_XHCI_THIS hub.op_regs.HcCrcr.crc |= (Bit64u) (value & ~0x3F);
}
#else
BX_XHCI_THIS hub.op_regs.HcCrcr.crc = (Bit64u) (value & ~0x3F);
#endif
BX_XHCI_THIS hub.ring_members.command_ring.dq_pointer = BX_XHCI_THIS hub.op_regs.HcCrcr.crc;
BX_XHCI_THIS hub.ring_members.command_ring.rcs = BX_XHCI_THIS hub.op_regs.HcCrcr.rcs;
// if command stop or abort, stop command ring
if (BX_XHCI_THIS hub.op_regs.HcCrcr.ca || BX_XHCI_THIS hub.op_regs.HcCrcr.cs)
BX_XHCI_THIS hub.op_regs.HcCrcr.crr = 0;
break;
case 0x1C: // Command Ring Control Register (Hi)
#if ADDR_CAP_64
BX_XHCI_THIS hub.op_regs.HcCrcr.crc &= (Bit64u) 0xFFFFFFFF;
BX_XHCI_THIS hub.op_regs.HcCrcr.crc |= (Bit64u) ((Bit64u) value << 32);
BX_XHCI_THIS hub.ring_members.command_ring.dq_pointer = BX_XHCI_THIS hub.op_regs.HcCrcr.crc;
#endif
break;
case 0x20: // Reserved and Zero'd
case 0x24: // Reserved and Zero'd
case 0x28: // Reserved and Zero'd
case 0x2C: // Reserved and Zero'd
if (value != 0)
BX_ERROR(("Write non-zero to RsvdZ Register (offset = 0x%08X value = 0x%08X)", offset, value));
break;
case 0x30: // DCBAAP (Lo)
if (value & 0x3F)
BX_ERROR(("Write non-zero to RsvdZ member of DCBAAP Register"));
#if ADDR_CAP_64
if (len == 8)
BX_XHCI_THIS hub.op_regs.HcDCBAAP.dcbaap = (Bit64u) (((Bit64u) value_hi << 32) | (value & ~0x3F));
else {
BX_XHCI_THIS hub.op_regs.HcDCBAAP.dcbaap &= (Bit64u) ~0xFFFFFFFF;
BX_XHCI_THIS hub.op_regs.HcDCBAAP.dcbaap |= (Bit64u) (value & ~0x3F);
}
#else
BX_XHCI_THIS hub.op_regs.HcDCBAAP.dcbaap = (Bit64u) (value & ~0x3F);
#endif
break;
case 0x34: // DCBAAP (Hi)
#if ADDR_CAP_64
BX_XHCI_THIS hub.op_regs.HcDCBAAP.dcbaap &= (Bit64u) 0xFFFFFFFF;
BX_XHCI_THIS hub.op_regs.HcDCBAAP.dcbaap |= (Bit64u) ((Bit64u) value << 32);
#endif
break;
case 0x38: // Config
temp = BX_XHCI_THIS hub.op_regs.HcConfig.RsvdP;
#if ((VERSION_MAJOR == 1) && (VERSION_MINOR >= 0x10))
BX_XHCI_THIS hub.op_regs.HcConfig.RsvdP = (value >> 10);
if (temp != BX_XHCI_THIS hub.op_regs.HcConfig.RsvdP)
BX_ERROR(("bits 31:10 in Config Register were not preserved"));
BX_XHCI_THIS hub.op_regs.HcConfig.cie = (value & (1 << 9)) ? 1 : 0;
BX_XHCI_THIS hub.op_regs.HcConfig.u3e = (value & (1 << 8)) ? 1 : 0;
#else
BX_XHCI_THIS hub.op_regs.HcConfig.RsvdP = (value >> 8);
if (temp != BX_XHCI_THIS hub.op_regs.HcConfig.RsvdP)
BX_ERROR(("bits 31:8 in Config Register were not preserved"));
#endif
BX_XHCI_THIS hub.op_regs.HcConfig.MaxSlotsEn = (value & 0xFF);
break;
}
} else
// Register Port Sets
if ((offset >= PORT_SET_OFFSET) && (offset < (PORT_SET_OFFSET + (USB_XHCI_PORTS * 16)))) {
unsigned port = (((offset - PORT_SET_OFFSET) >> 4) & 0x3F); // calculate port number
switch (offset & 0x0000000F) {
case 0x00:
if (value & (1<<9)) { // port power
if (value & ((1<<30) | (1<<24) | (1<<3) | (1<<0)))
BX_DEBUG(("Write to one or more Read-only bits in PORTSC[%i] Register (0x%08X)", port, value));
if (value & ((3<<28) | (1<<2)))
BX_ERROR(("Write non-zero to a RsvdZ member of PORTSC[%i] Register", port));
if (BX_XHCI_THIS hub.usb_port[port].is_usb3) {
BX_XHCI_THIS hub.usb_port[port].portsc.wpr = (value & (1 << 31)) ? 1 : 0;
BX_XHCI_THIS hub.usb_port[port].portsc.cec = (value & (1 << 23)) ? 1 : 0;
BX_XHCI_THIS hub.usb_port[port].portsc.wrc = (value & (1 << 19)) ? 0 : BX_XHCI_THIS hub.usb_port[port].portsc.wrc;
if (value & (1<<18))
BX_ERROR(("Write to USB3 port: bit 18"));
} else {
BX_XHCI_THIS hub.usb_port[port].portsc.pec = (value & (1 << 18)) ? 0 : BX_XHCI_THIS hub.usb_port[port].portsc.pec;
if (value & ((1<<31) | (1<<23) | (1<<19)))
BX_ERROR(("Write to USB2 port: RsvdZ bit"));
}
// The WC1 bits must be before anything that will change these bits below.
if (value & (1 << 22)) BX_XHCI_THIS hub.usb_port[port].portsc.plc = 0;
if (value & (1 << 21)) BX_XHCI_THIS hub.usb_port[port].portsc.prc = 0;
if (value & (1 << 20)) BX_XHCI_THIS hub.usb_port[port].portsc.occ = 0;
if (value & (1 << 17)) BX_XHCI_THIS hub.usb_port[port].portsc.csc = 0;
if (value & (1 << 1)) BX_XHCI_THIS hub.usb_port[port].portsc.ped = 0;
BX_XHCI_THIS hub.usb_port[port].portsc.woe = (value & (1 << 27)) ? 1 : 0;
BX_XHCI_THIS hub.usb_port[port].portsc.wde = (value & (1 << 26)) ? 1 : 0;
BX_XHCI_THIS hub.usb_port[port].portsc.wce = (value & (1 << 25)) ? 1 : 0;
BX_XHCI_THIS hub.usb_port[port].portsc.pic = (value & (0x3 << 14)) >> 14;
// if transition from non-powered to powered
if (BX_XHCI_THIS hub.usb_port[port].portsc.pp == 0) {
// a "has been reset" is false.
BX_XHCI_THIS hub.usb_port[port].has_been_reset = 0;
}
BX_XHCI_THIS hub.usb_port[port].portsc.pp = 1;
if (value & (1<<16)) { // LWS
switch ((value & (0xF << 5)) >> 5) {
case 0:
BX_XHCI_THIS hub.usb_port[port].portsc.pls = PLS_U0;
break;
case 2: // USB2 only
BX_XHCI_THIS hub.usb_port[port].portsc.pls = PLS_U2;
break;
case 3:
BX_XHCI_THIS hub.usb_port[port].portsc.pls = PLS_U3_SUSPENDED;
break;
case 5: // USB3 only
if (BX_XHCI_THIS hub.usb_port[port].portsc.pls == PLS_DISABLED) {
BX_XHCI_THIS hub.usb_port[port].portsc.pls = PLS_RXDETECT;
BX_XHCI_THIS hub.usb_port[port].portsc.ped = 0;
BX_XHCI_THIS hub.usb_port[port].portsc.pec = 1;
}
break;
case 15: // USB2 only
if (BX_XHCI_THIS hub.usb_port[port].portsc.pls == PLS_U3_SUSPENDED) {
// port should transition to the U3Exit state...
}
break;
}
}
// if port reset bit is set, reset the port, then enable the port (if ccs == 1).
if (((value & (1 << 31)) && BX_XHCI_THIS hub.usb_port[port].is_usb3) ||
(value & (1 << 4))) {
reset_port_usb3(port, (value & (1 << 4)) ? HOT_RESET : WARM_RESET);
}
} else
BX_XHCI_THIS hub.usb_port[port].portsc.pp = 0;
break;
case 0x04:
if (BX_XHCI_THIS hub.usb_port[port].portsc.pp) {
if (BX_XHCI_THIS hub.usb_port[port].is_usb3) {
temp = BX_XHCI_THIS hub.usb_port[port].usb3.portpmsc.RsvdP;
BX_XHCI_THIS hub.usb_port[port].usb3.portpmsc.RsvdP = value >> 17;
if (temp != BX_XHCI_THIS hub.usb_port[port].usb3.portpmsc.RsvdP)
BX_ERROR(("bits 31:17 in PORTPMSC[%i] Register were not preserved", port));
BX_XHCI_THIS hub.usb_port[port].usb3.portpmsc.fla = (value & (1 << 16)) ? 1 : 0;
BX_XHCI_THIS hub.usb_port[port].usb3.portpmsc.u2timeout = (value & (0xFF << 8)) >> 8;
BX_XHCI_THIS hub.usb_port[port].usb3.portpmsc.u1timeout = (value & (0xFF << 0)) >> 0;
} else {
if (value & (7<<0))
BX_ERROR(("Write to one or more Read-only bits in PORTPMSC[%i] Register", port));
BX_XHCI_THIS hub.usb_port[port].usb2.portpmsc.tmode = (value & (0xF << 28)) >> 28;
temp = BX_XHCI_THIS hub.usb_port[port].usb2.portpmsc.RsvdP;
BX_XHCI_THIS hub.usb_port[port].usb2.portpmsc.RsvdP = (value & (0x1FFF << 15)) >> 15;
if (temp != BX_XHCI_THIS hub.usb_port[port].usb2.portpmsc.RsvdP)
BX_ERROR(("bits 27:15 in PORTPMSC[%i] Register were not preserved", port));
BX_XHCI_THIS hub.usb_port[port].usb2.portpmsc.hle = (value & (1 << 16)) ? 1 : 0;
BX_XHCI_THIS hub.usb_port[port].usb2.portpmsc.l1dslot = (value & (0xFF << 8)) >> 8;
BX_XHCI_THIS hub.usb_port[port].usb2.portpmsc.hird = (value & (0xF << 4)) >> 4;
BX_XHCI_THIS hub.usb_port[port].usb2.portpmsc.rwe = (value & (1 << 3)) ? 1 : 0;
}
}
break;
case 0x08:
if (BX_XHCI_THIS hub.usb_port[port].portsc.pp) {
if (BX_XHCI_THIS hub.usb_port[port].is_usb3) {
if (value & (0xFFFF<<0))
BX_ERROR(("Write to one or more Read-only bits in PORTLI[%i] Register", port));
temp = BX_XHCI_THIS hub.usb_port[port].usb3.portli.RsvdP;
BX_XHCI_THIS hub.usb_port[port].usb3.portli.RsvdP = value >> 16;
if (temp != BX_XHCI_THIS hub.usb_port[port].usb3.portli.RsvdP)
BX_ERROR(("bits 31:16 in PORTLI[%i] Register were not preserved", port));
} else {
temp = BX_XHCI_THIS hub.usb_port[port].usb2.portli.RsvdP;
BX_XHCI_THIS hub.usb_port[port].usb2.portli.RsvdP = value;
if (temp != BX_XHCI_THIS hub.usb_port[port].usb2.portli.RsvdP)
BX_ERROR(("bits 31:0 in PORTLI[%i] Register were not preserved", port));
}
}
break;
case 0x0C:
#if ((VERSION_MAJOR < 1) || ((VERSION_MAJOR == 1) && (VERSION_MINOR == 0)))
BX_ERROR(("Write to Reserved Register in Port Register Set %i", port));
#else
temp = BX_XHCI_THIS hub.usb_port[port].porthlpmc.RsvdP;
BX_XHCI_THIS hub.usb_port[port].porthlpmc.RsvdP = (value >> 14);
if (temp != BX_XHCI_THIS hub.usb_port[port].porthlpmc.RsvdP)
BX_ERROR(("bits 31:14 in PORTHLPMC[%i] Register were not preserved", port));
BX_XHCI_THIS hub.usb_port[port].porthlpmc.hirdd = ((value & (0x0F << 10)) >> 10);
BX_XHCI_THIS hub.usb_port[port].porthlpmc.l1timeout = ((value & (0xFF << 2)) >> 2);
BX_XHCI_THIS hub.usb_port[port].porthlpmc.hirdm = ((value & (0x0F << 0)) >> 0);
#endif
break;
}
} else
// Extended Capabilities
if ((offset >= EXT_CAPS_OFFSET) && (offset < (EXT_CAPS_OFFSET + EXT_CAPS_SIZE))) {
unsigned caps_offset = (offset - EXT_CAPS_OFFSET);
Bit64u qword = (((Bit64u) value_hi << 32) | value);
while (len) {
*(Bit8u *) &BX_XHCI_THIS hub.extended_caps[caps_offset] = (Bit8u) (qword & 0xFF);
switch (caps_offset) {
case 3:
if (qword & 1) // clear the BIOS owner bit
BX_XHCI_THIS hub.extended_caps[2] &= ~(1<<0);
break;
default:
;
}
len--;
caps_offset++;
qword >>= 8;
}
} else
// Host Controller Runtime Registers
if ((offset >= RUNTIME_OFFSET) && (offset < (RUNTIME_OFFSET + 32 + (INTERRUPTERS * 32)))) {
if (offset == RUNTIME_OFFSET) {
BX_ERROR(("Write to MFINDEX register"));
} else if (offset < (RUNTIME_OFFSET + 32)) {
BX_ERROR(("Write to Reserved Register in HC Runtime Register set"));
} else {
unsigned rt_offset = (offset - RUNTIME_OFFSET - 32);
i = (rt_offset >> 5); // interrupter offset
switch (rt_offset & 0x1F) {
case 0x00:
temp = BX_XHCI_THIS hub.runtime_regs.interrupter[i].iman.RsvdP;
BX_XHCI_THIS hub.runtime_regs.interrupter[i].iman.RsvdP = (value >> 2);
if (temp != BX_XHCI_THIS hub.runtime_regs.interrupter[i].iman.RsvdP)
BX_ERROR(("bits 31:2 in IMAN Register were not preserved"));
BX_XHCI_THIS hub.runtime_regs.interrupter[i].iman.ie = ((value & (1 << 1)) == (1 << 1));
if (value & (1 << 0))
BX_XHCI_THIS hub.runtime_regs.interrupter[i].iman.ip = 0;
break;
case 0x04:
BX_XHCI_THIS hub.runtime_regs.interrupter[i].imod.imodc = (value >> 16);
BX_XHCI_THIS hub.runtime_regs.interrupter[i].imod.imodi = (value & 0xFFFF);
break;
case 0x08:
temp = BX_XHCI_THIS hub.runtime_regs.interrupter[i].erstsz.RsvdP;
BX_XHCI_THIS hub.runtime_regs.interrupter[i].erstsz.RsvdP = (value >> 16);
if (temp != BX_XHCI_THIS hub.runtime_regs.interrupter[i].erstsz.RsvdP)
BX_ERROR(("bits 31:16 in ERSTSZ Register were not preserved"));
BX_XHCI_THIS hub.runtime_regs.interrupter[i].erstsz.erstabsize = (value & 0xFFFF);
break;
case 0x0C:
temp = BX_XHCI_THIS hub.runtime_regs.interrupter[i].RsvdP;
BX_XHCI_THIS hub.runtime_regs.interrupter[i].RsvdP = value;
if (temp != BX_XHCI_THIS hub.runtime_regs.interrupter[i].RsvdP)
BX_ERROR(("bits 31:0 in RsvdP (0x0C) Register were not preserved"));
break;
case 0x10:
temp = BX_XHCI_THIS hub.runtime_regs.interrupter[i].erstba.RsvdP;
BX_XHCI_THIS hub.runtime_regs.interrupter[i].erstba.RsvdP = (value & ERSTABADD_MASK);
if (temp != BX_XHCI_THIS hub.runtime_regs.interrupter[i].erstba.RsvdP)
BX_ERROR(("RsvdP bits in ERSTBA Register were not preserved"));
#if ADDR_CAP_64
if (len == 8) {
BX_XHCI_THIS hub.runtime_regs.interrupter[i].erstba.erstabadd = (((Bit64u) value_hi << 32) | (value & ~ERSTABADD_MASK));
init_event_ring(i); // initialize event ring members
} else {
BX_XHCI_THIS hub.runtime_regs.interrupter[i].erstba.erstabadd &= (Bit64u) ~0xFFFFFFFF;
BX_XHCI_THIS hub.runtime_regs.interrupter[i].erstba.erstabadd |= (Bit64u) (value & ~ERSTABADD_MASK);
}
#else
BX_XHCI_THIS hub.runtime_regs.interrupter[i].erstba.erstabadd = (Bit64u) (value & ~ERSTABADD_MASK);
init_event_ring(i); // initialize event ring members
#endif
break;
case 0x14:
#if ADDR_CAP_64
BX_XHCI_THIS hub.runtime_regs.interrupter[i].erstba.erstabadd &= (Bit64u) 0xFFFFFFFF;
BX_XHCI_THIS hub.runtime_regs.interrupter[i].erstba.erstabadd |= ((Bit64u) value << 32);
init_event_ring(i); // initialize event ring members
#endif
break;
case 0x18:
if (value & (1 << 3))
BX_XHCI_THIS hub.runtime_regs.interrupter[i].erdp.ehb = 0;
BX_XHCI_THIS hub.runtime_regs.interrupter[i].erdp.desi = (value & 0x07);
#if ADDR_CAP_64
if (len == 8)
BX_XHCI_THIS hub.runtime_regs.interrupter[i].erdp.eventadd = (Bit64u) (((Bit64u) value_hi << 32) | (value & ~0x0F));
else {
BX_XHCI_THIS hub.runtime_regs.interrupter[i].erdp.eventadd &= (Bit64u) ~0xFFFFFFFF;
BX_XHCI_THIS hub.runtime_regs.interrupter[i].erdp.eventadd |= (Bit64u) (value & ~0x0F);
}
#else
BX_XHCI_THIS hub.runtime_regs.interrupter[i].erdp.eventadd == (Bit64u) (value & ~0x0F);
#endif
break;
case 0x1C:
#if ADDR_CAP_64
BX_XHCI_THIS hub.runtime_regs.interrupter[i].erdp.eventadd &= (Bit64u) 0xFFFFFFFF;
BX_XHCI_THIS hub.runtime_regs.interrupter[i].erdp.eventadd |= ((Bit64u) value << 32);
#endif
break;
}
}
} else
// Doorbell Registers
if ((offset >= DOORBELL_OFFSET) && (offset < (DOORBELL_OFFSET + 4 + (INTERRUPTERS * 4)))) {
if (value & (0xFF << 8))
BX_ERROR(("RsvdZ field of Doorbell written as non zero."));
unsigned doorbell = ((offset - DOORBELL_OFFSET) >> 2);
if (doorbell == 0) { // Command Doorbell
BX_DEBUG(("Command Doorbell Rang"));
if (value & (0xFFFF << 16))
BX_ERROR(("DB Stream ID not zero when Command Doorbell rung"));
if ((value & 0xFF) != 0)
BX_ERROR(("Command Doorbell rang with non zero value: 0x%02X", (value & 0xFF)));
// if the run/stop bit is set, the command ring is running
if (BX_XHCI_THIS hub.op_regs.HcCommand.rs)
BX_XHCI_THIS hub.op_regs.HcCrcr.crr = 1;
process_command_ring();
} else {
// doorbell = slot to use (1 based)
// (value & 0xFF) = ep (1 = control, 2 = ep1 out, 3 = ep1 in, etc);
int ep = (value & 0xFF);
BX_DEBUG(("Rang Doorbell: slot = %i ep = %i (%s)", doorbell, ep, (ep & 1) ? "IN" : "OUT"));
if (ep > 31)
BX_ERROR(("Doorbell rang with EP > 31 (ep = %i)", ep));
else
process_transfer_ring(doorbell, ep);
}
} else
BX_ERROR(("register write to unknown offset 0x%08X: 0x%08X%08X (len=%i)", offset, (Bit32u) value_hi, (Bit32u) value, len));
return 1;
}
void xhci_event_handler(int event, USBPacket *packet, void *dev, int port)
{
((bx_usb_xhci_c*)dev)->event_handler(event, packet, port);
}
void bx_usb_xhci_c::event_handler(int event, USBPacket *packet, int port)
{
int slot, ep;
if (event == USB_EVENT_ASYNC) {
BX_DEBUG(("Experimental async packet completion"));
USBAsync *p = container_of_usb_packet(packet);
p->done = 1;
slot = (p->slot_ep >> 8);
ep = (p->slot_ep & 0xff);
BX_XHCI_THIS process_transfer_ring(slot, ep);
} else if (event == USB_EVENT_WAKEUP) {
if (BX_XHCI_THIS hub.usb_port[port].portsc.pls != PLS_U3_SUSPENDED) {
return;
}
BX_XHCI_THIS hub.usb_port[port].portsc.pls = PLS_RESUME;
if (!BX_XHCI_THIS hub.usb_port[port].portsc.plc) {
BX_XHCI_THIS hub.usb_port[port].portsc.plc = 1;
if (BX_XHCI_THIS hub.op_regs.HcStatus.hch) {
return;
}
write_event_TRB(0, ((port + 1) << 24), TRB_SET_COMP_CODE(1), TRB_SET_TYPE(PORT_STATUS_CHANGE), 1);
}
} else {
BX_ERROR(("unknown/unsupported event (id=%d) on port #%d", event, port+1));
}
}
// This function checks and processes all enqueued TRB's in the EP's transfer ring
void bx_usb_xhci_c::process_transfer_ring(const int slot, const int ep)
{
struct TRB trb;
Bit64u address = 0, org_addr;
int int_target/*, td_size*/;
Bit32u transfer_length;
int ret = 0, len;
int port_num = BX_XHCI_THIS hub.slots[slot].slot_context.rh_port_num;
USBAsync *p;
Bit8u cur_direction = (ep & 1) ? USB_TOKEN_IN : USB_TOKEN_OUT; // for NORMAL without SETUP
bool is_transfer_trb, is_immed_data, ioc, spd_occurred = 0;
bool first_event_trb_encountered = 0;
Bit32u bytes_not_transferred = 0;
int comp_code = 0;
Bit8u immed_data[8];
// this assumes that we are starting at the first of the TD when this function is called.
// this is usually the case, and rarely isn't.
int trb_count = 0;
BX_XHCI_THIS hub.slots[slot].ep_context[ep].edtla = 0;
BX_XHCI_THIS hub.slots[slot].ep_context[ep].retry = 0;
// if the ep is disabled, return an error event trb.
if ((BX_XHCI_THIS hub.slots[slot].slot_context.slot_state == SLOT_STATE_DISABLED_ENABLED)
|| (BX_XHCI_THIS hub.slots[slot].ep_context[ep].ep_context.ep_state == EP_STATE_DISABLED)) {
org_addr = BX_XHCI_THIS hub.slots[slot].ep_context[ep].enqueue_pointer;
write_event_TRB(0, org_addr, TRB_SET_COMP_CODE(EP_NOT_ENABLED),
TRB_SET_SLOT(slot) | TRB_SET_EP(ep) | TRB_SET_TYPE(TRANS_EVENT), 1);
return;
}
// if the ep is in the halted or error state, ignore the doorbell ring.
if ((BX_XHCI_THIS hub.slots[slot].ep_context[ep].ep_context.ep_state == EP_STATE_HALTED)
|| (BX_XHCI_THIS hub.slots[slot].ep_context[ep].ep_context.ep_state == EP_STATE_ERROR))
return;
// if the ep_context::type::direction field is not correct for the ep type of this ep, then ignore the doorbell
if (ep >= 2) {
static int endpoint_dir[8] = { -1, EP_DIR_OUT, EP_DIR_OUT, EP_DIR_OUT, -1, EP_DIR_IN, EP_DIR_IN, EP_DIR_IN };
int ep_type = (ep & 1) ? EP_DIR_IN : EP_DIR_OUT;
if (endpoint_dir[BX_XHCI_THIS hub.slots[slot].ep_context[ep].ep_context.ep_type] != ep_type) {
BX_ERROR(("Endpoint_context::Endpoint_type::direction is not correct for this endpoint number. Ignoring doorbell ring."));
return;
}
}
// If an endpoint is in the Stopped state when the doorbell is rung, it will transition to the Running state (page 126)
// The output Context (*slot_addr) should be updated before any other transfer events are made
if (BX_XHCI_THIS hub.slots[slot].ep_context[ep].ep_context.ep_state == EP_STATE_STOPPED) {
BX_XHCI_THIS hub.slots[slot].ep_context[ep].ep_context.ep_state = EP_STATE_RUNNING;
update_ep_context(slot, ep);
}
// read in the TRB
read_TRB((bx_phy_address) BX_XHCI_THIS hub.slots[slot].ep_context[ep].enqueue_pointer, &trb);
BX_DEBUG(("Found TRB: address = 0x" FORMATADDRESS " 0x" FMT_ADDRX64 " 0x%08X 0x%08X %i",
(bx_phy_address) BX_XHCI_THIS hub.slots[slot].ep_context[ep].enqueue_pointer,
trb.parameter, trb.status, trb.command, BX_XHCI_THIS hub.slots[slot].ep_context[ep].rcs));
while ((trb.command & 1) == BX_XHCI_THIS hub.slots[slot].ep_context[ep].rcs) {
org_addr = BX_XHCI_THIS hub.slots[slot].ep_context[ep].enqueue_pointer;
BX_DEBUG(("Found TRB: address = 0x" FORMATADDRESS " 0x" FMT_ADDRX64 " 0x%08X 0x%08X %i (SPD occurred = %i)",
(bx_phy_address) org_addr, trb.parameter, trb.status, trb.command, BX_XHCI_THIS hub.slots[slot].ep_context[ep].rcs, spd_occurred));
trb_count++;
// these are used in some/most items.
// If not used, won't hurt to extract bad data.
int_target = TRB_GET_TARGET(trb.status);
// td_size = TRB_GET_TDSIZE(trb.status);
transfer_length = TRB_GET_TX_LEN(trb.status);
is_transfer_trb = 0; // assume not a transfer
ioc = TRB_IOC(trb.command);
// if a SPD occurred, we only process the LINK and EVENT TRB's in this TD, until either on of these two or end of TD
if (!spd_occurred ||
(spd_occurred && ((TRB_GET_TYPE(trb.command) == LINK) || (TRB_GET_TYPE(trb.command) == EVENT_DATA)))) {
// is the data in trb.parameter? (Immediate data?)
is_immed_data = TRB_IS_IMMED_DATA(trb.command);
if (is_immed_data)
DEV_MEM_READ_PHYSICAL_DMA((bx_phy_address) org_addr, 8, immed_data); // No byte-swapping here
else
address = trb.parameter;
switch (TRB_GET_TYPE(trb.command)) {
// is a LINK trb.
case LINK:
if (ioc)
write_event_TRB(int_target, org_addr, TRB_SET_COMP_CODE(TRB_SUCCESS), TRB_SET_TYPE(LINK), 1);
if (TRB_TOGGLE(trb.command))
BX_XHCI_THIS hub.slots[slot].ep_context[ep].rcs ^= 1;
BX_XHCI_THIS hub.slots[slot].ep_context[ep].enqueue_pointer = trb.parameter & (Bit64u) ~0xF;
BX_DEBUG(("0x" FORMATADDRESS ": Transfer Ring (slot = %i) (ep = %i): LINK TRB: New dq_pointer = 0x" FMT_ADDRX64 " (%i)",
(bx_phy_address) org_addr, slot, ep, BX_XHCI_THIS hub.slots[slot].ep_context[ep].enqueue_pointer, BX_XHCI_THIS hub.slots[slot].ep_context[ep].rcs));
#if ((VERSION_MAJOR == 0) && (VERSION_MINOR == 0x95))
// https://patchwork.kernel.org/patch/51191/
if (!TRB_CHAIN(trb.command))
BX_DEBUG(("Chain Bit in Link TRB not set."));
#endif
read_TRB((bx_phy_address) BX_XHCI_THIS hub.slots[slot].ep_context[ep].enqueue_pointer, &trb);
continue;
// Setup Stage TRB
case SETUP_STAGE:
cur_direction = USB_TOKEN_SETUP;
is_transfer_trb = 1;
BX_DEBUG(("0x" FORMATADDRESS ": Transfer Ring (slot = %i) (ep = %i) (len = %i): Found SETUP TRB",
(bx_phy_address) org_addr, slot, ep, transfer_length));
break;
// Data Stage TRB
case DATA_STAGE:
cur_direction = TRB_GET_DIR(trb.command);
is_transfer_trb = 1;
BX_DEBUG(("0x" FORMATADDRESS ": Transfer Ring (slot = %i) (ep = %i) (len = %i): Found DATA STAGE TRB",
(bx_phy_address) org_addr, slot, ep, transfer_length));
break;
// Status Stage TRB
case STATUS_STAGE:
cur_direction = TRB_GET_DIR(trb.command);
is_transfer_trb = 1;
transfer_length = 0;
BX_DEBUG(("0x" FORMATADDRESS ": Transfer Ring (slot = %i) (ep = %i): Found STATUS STAGE TRB",
(bx_phy_address) org_addr, slot, ep));
break;
// Normal TRB
case NORMAL:
is_transfer_trb = 1;
BX_DEBUG(("0x" FORMATADDRESS ": Transfer Ring (slot = %i) (ep = %i) (len = %i): Found NORMAL TRB",
(bx_phy_address) org_addr, slot, ep, transfer_length));
break;
// Event TRB
// xHCI version 1.10 (Nov 2017), page 184
case EVENT_DATA:
if (!spd_occurred || (spd_occurred && !first_event_trb_encountered)) {
comp_code = (spd_occurred) ? SHORT_PACKET : TRB_SUCCESS;
write_event_TRB(int_target, trb.parameter,
TRB_SET_COMP_CODE(comp_code) | (BX_XHCI_THIS hub.slots[slot].ep_context[ep].edtla & 0x00FFFFFF),
TRB_SET_SLOT(slot) | TRB_SET_EP(ep) | TRB_SET_TYPE(TRANS_EVENT) | (1<<2),
ioc);
}
// in version 1.00+, we need to check the PARSE_ALL_EVENT bit
// to see if we parse all of them
#if (((VERSION_MAJOR == 1) && (VERSION_MINOR >= 0x00)) && PARSE_ALL_EVENT)
first_event_trb_encountered = 0;
#else
if (spd_occurred)
first_event_trb_encountered = 1;
#endif
BX_DEBUG(("0x" FORMATADDRESS ": Transfer Ring (slot = %i) (ep = %i) (trnsfrd = %i): Found EVENT_DATA TRB: (returning %i)",
(bx_phy_address) org_addr, slot, ep, BX_XHCI_THIS hub.slots[slot].ep_context[ep].edtla, comp_code));
break;
// No Op (transfer ring) TRB (Sect: 6.4.1.4)
case NO_OP:
BX_DEBUG(("0x" FORMATADDRESS ": Transfer Ring (slot = %i) (ep = %i): Found No Op TRB",
(bx_phy_address) org_addr, slot, ep));
cur_direction = 0;
is_transfer_trb = 1;
transfer_length = 0;
break;
// unknown TRB type
default:
BX_ERROR(("0x" FORMATADDRESS ": Transfer Ring (slot = %i) (ep = %i): Unknown TRB found.",
(bx_phy_address) org_addr, slot, ep));
BX_ERROR(("Unknown trb type found: %i(dec) (0x" FMT_ADDRX64 " 0x%08X 0x%08X)", TRB_GET_TYPE(trb.command),
trb.parameter, trb.status, trb.command));
}
// is there a transfer to be done?
if (is_transfer_trb) {
p = find_async_packet(&BX_XHCI_THIS packets, org_addr);
bool completion = (p != NULL);
if (completion && !p->done) {
return;
}
comp_code = TRB_SUCCESS; // assume good trans event
if (completion) {
ret = p->packet.len;
len = ret;
} else {
p = create_async_packet(&BX_XHCI_THIS packets, org_addr, transfer_length);
p->packet.pid = cur_direction;
p->packet.devaddr = BX_XHCI_THIS hub.slots[slot].slot_context.device_address;
p->packet.devep = (ep >> 1);
p->packet.complete_cb = xhci_event_handler;
p->packet.complete_dev = BX_XHCI_THIS_PTR;
p->slot_ep = (slot << 8) | ep;
switch (cur_direction) {
case USB_TOKEN_OUT:
case USB_TOKEN_SETUP:
if (is_immed_data)
memcpy(p->packet.data, immed_data, transfer_length);
else if (transfer_length > 0)
DEV_MEM_READ_PHYSICAL_DMA((bx_phy_address) address, transfer_length, p->packet.data);
// The XHCI should block all SET_ADDRESS SETUP TOKEN's
if ((cur_direction == USB_TOKEN_SETUP) &&
(p->packet.data[0] == 0) && // Request type
(p->packet.data[1] == 5)) { // SET_ADDRESS
len = 0;
comp_code = TRB_ERROR;
BX_ERROR(("SETUP_TOKEN: System Software should not send SET_ADDRESS command on the xHCI."));
} else {
ret = BX_XHCI_THIS broadcast_packet(&p->packet, port_num - 1);
len = transfer_length;
BX_DEBUG(("OUT: Transferred %i bytes (ret = %i)", len, ret));
}
break;
case USB_TOKEN_IN:
ret = BX_XHCI_THIS broadcast_packet(&p->packet, port_num - 1);
break;
}
if (ret == USB_RET_ASYNC) {
BX_DEBUG(("Async packet deferred"));
break;
}
}
if (cur_direction == USB_TOKEN_IN) {
if (ret >= 0) {
len = ret;
BX_XHCI_THIS hub.slots[slot].ep_context[ep].edtla += len;
if (len > 0)
DEV_MEM_WRITE_PHYSICAL_DMA((bx_phy_address) address, len, p->packet.data);
BX_DEBUG(("IN: Transferred %i bytes, requested %i bytes", len, transfer_length));
if (len < (int) transfer_length) {
bytes_not_transferred = transfer_length - len;
spd_occurred = 1;
} else
bytes_not_transferred = 0;
} else {
switch (ret) {
case USB_RET_STALL:
comp_code = STALL_ERROR;
break;
case USB_RET_BABBLE:
comp_code = BABBLE_DETECTION;
break;
default:
comp_code = TRANSACTION_ERROR;
}
len = 0;
}
}
remove_async_packet(&BX_XHCI_THIS packets, p);
if (ret == USB_RET_NAK) {
// If we are a high-speed device, and a SETUP packet, we need to
// return a comp_code of TRANSACTION_ERROR instead of breaking out.
if ((cur_direction == USB_TOKEN_SETUP) &&
(BX_XHCI_THIS hub.slots[slot].slot_context.speed == SPEED_HI))
comp_code = TRANSACTION_ERROR;
else {
// if the device NAK'ed, we retry with given interval
BX_XHCI_THIS hub.slots[slot].ep_context[ep].retry = 1;
int interval = 125 * (1 << BX_XHCI_THIS hub.slots[slot].ep_context[ep].ep_context.interval);
if (interval < 1000) {
BX_XHCI_THIS hub.slots[slot].ep_context[ep].retry_counter = 1;
} else {
BX_XHCI_THIS hub.slots[slot].ep_context[ep].retry_counter = interval / 1000;
}
break;
}
}
// 4.10.1 paragraph 4
// 4.10.1.1
if (ioc) {
if ((comp_code == TRB_SUCCESS) && spd_occurred && TRB_SPD(trb.command)) {
comp_code = SHORT_PACKET;
BX_DEBUG(("Sending Short Packet Detect Event TRB (%i)", bytes_not_transferred));
}
// create a Event TRB
write_event_TRB(int_target, org_addr, TRB_SET_COMP_CODE(comp_code) | bytes_not_transferred,
TRB_SET_SLOT(slot) | TRB_SET_EP(ep) | TRB_SET_TYPE(TRANS_EVENT), 1);
}
}
}
// if the Chain bit is clear, then end of TD
if ((trb.command & (1<<4)) == 0) {
spd_occurred = 0;
BX_XHCI_THIS hub.slots[slot].ep_context[ep].edtla = 0;
}
// advance the Dequeue pointer and continue;
BX_XHCI_THIS hub.slots[slot].ep_context[ep].enqueue_pointer += 16;
read_TRB((bx_phy_address) BX_XHCI_THIS hub.slots[slot].ep_context[ep].enqueue_pointer, &trb);
}
BX_DEBUG(("Process Transfer Ring: Processed %i TRB's", trb_count));
if (trb_count == 0)
BX_ERROR(("Process Transfer Ring: Doorbell rang, but no TRB's were enqueued in the ring."));
}
// This function call starts at the current position in the Command Ring,
// processes that command, then moves to the next one, until CCS != SCS;
// BX_XHCI_THIS hub.ring_members.command_ring.dq_pointer = current position in ring
// BX_XHCI_THIS hub.op_regs.HcCrcr.rcs = current ring cycle state
// BX_XHCI_THIS hub.op_regs.HcCrcr.crr = command ring running
void bx_usb_xhci_c::process_command_ring(void)
{
struct TRB trb;
int i, slot, /*slot_type,*/ ep, comp_code = 0, new_addr = 0, bsr = 0;
Bit32u a_flags = 0, d_flags, tmpval1, tmpval2;
Bit64u org_addr;
Bit8u buffer[CONTEXT_SIZE + (32 * CONTEXT_SIZE)];
struct SLOT_CONTEXT slot_context;
struct EP_CONTEXT ep_context;
if (!BX_XHCI_THIS hub.op_regs.HcCrcr.crr)
return;
// read in the TRB
read_TRB((bx_phy_address) BX_XHCI_THIS hub.ring_members.command_ring.dq_pointer, &trb);
BX_DEBUG(("Dump command trb: %i(dec) (0x" FMT_ADDRX64 " 0x%08X 0x%08X) (%i)", TRB_GET_TYPE(trb.command),
trb.parameter, trb.status, trb.command, BX_XHCI_THIS hub.ring_members.command_ring.rcs));
while ((trb.command & 1) == BX_XHCI_THIS hub.ring_members.command_ring.rcs) {
org_addr = BX_XHCI_THIS hub.ring_members.command_ring.dq_pointer;
switch (TRB_GET_TYPE(trb.command)) {
// is a LINK trb.
case LINK:
// Chain bit and Interrupter Target fields are ignored in Command Rings (Page 370)
BX_XHCI_THIS hub.ring_members.command_ring.dq_pointer = trb.parameter & (Bit64u) ~0xF;
if (TRB_IOC(trb.command))
write_event_TRB(0, org_addr, TRB_SET_COMP_CODE(TRB_SUCCESS), TRB_SET_SLOT(0) | TRB_SET_TYPE(LINK), 1);
if (TRB_TOGGLE(trb.command))
BX_XHCI_THIS hub.ring_members.command_ring.rcs ^= 1;
BX_DEBUG(("0x" FORMATADDRESS ": Command Ring: Found LINK TRB: New dq_pointer = 0x" FORMATADDRESS " (%i)",
(bx_phy_address) org_addr, (bx_phy_address) BX_XHCI_THIS hub.ring_members.command_ring.dq_pointer,
BX_XHCI_THIS hub.ring_members.command_ring.rcs));
read_TRB((bx_phy_address) BX_XHCI_THIS hub.ring_members.command_ring.dq_pointer, &trb);
continue;
// NEC: Get Firmware version
case NEC_TRB_TYPE_GET_FW:
write_event_TRB(0, org_addr, TRB_SET_COMP_CODE(1) | 0x3021, TRB_SET_SLOT(0) | TRB_SET_TYPE(NEC_TRB_TYPE_CMD_COMP), 1);
BX_INFO(("NEC GET Firmware Version TRB found. Returning 0x3021"));
break;
/* NEC: Get Verification
* The NEC/Renesas driver sends a vendor specific TRB to the controller to
* verify that the controller is indeed a NEC/Renesas controller. I do not
* have permission from NEC/Renesas to show the code that returns the correct
* verification values. Therefore, this simply returns. As long as you are
* not using a Windows driver, everything will be fine.
*/
case NEC_TRB_TYPE_GET_UN:
BX_INFO(("NEC GET Verification TRB found."));
break;
case ENABLE_SLOT:
comp_code = NO_SLOTS_ERROR; // assume no slots
slot = 0;
// slot_type = TRB_GET_STYPE(trb.command); // currently not used
for (i=1; i<MAX_SLOTS; i++) { // slots are one based
if (BX_XHCI_THIS hub.slots[i].enabled == 0) {
memset(&BX_XHCI_THIS hub.slots[i], 0, sizeof(struct HC_SLOT_CONTEXT));
BX_XHCI_THIS hub.slots[i].slot_context.slot_state = SLOT_STATE_DISABLED_ENABLED;
BX_XHCI_THIS hub.slots[i].enabled = 1;
slot = i;
for (i=1; i<32; i++)
BX_XHCI_THIS hub.slots[slot].ep_context[i].ep_context.ep_state = EP_STATE_DISABLED;
comp_code = TRB_SUCCESS;
break;
}
}
write_event_TRB(0, org_addr, TRB_SET_COMP_CODE(comp_code), TRB_SET_SLOT(slot) | TRB_SET_TYPE(COMMAND_COMPLETION), 1);
BX_DEBUG(("0x" FORMATADDRESS ": Command Ring: Found Enable Slot TRB (slot = %i) (returning %i)",
(bx_phy_address) org_addr, slot, comp_code));
break;
case DISABLE_SLOT:
slot = TRB_GET_SLOT(trb.command); // slots are 1 based
if (BX_XHCI_THIS hub.slots[slot].enabled) {
BX_XHCI_THIS hub.slots[slot].enabled = 0;
BX_XHCI_THIS hub.slots[slot].slot_context.slot_state = 0; // disabled
update_slot_context(slot);
for (i=1; i<32; i++) {
BX_XHCI_THIS hub.slots[slot].ep_context[i].ep_context.ep_state = EP_STATE_DISABLED;
update_ep_context(slot, i);
}
comp_code = TRB_SUCCESS;
} else
comp_code = SLOT_NOT_ENABLED;
write_event_TRB(0, org_addr, TRB_SET_COMP_CODE(comp_code), TRB_SET_SLOT(slot) | TRB_SET_TYPE(COMMAND_COMPLETION), 1);
BX_DEBUG(("0x" FORMATADDRESS ": Command Ring: Found Disable Slot TRB (slot = %i) (returning %i)",
(bx_phy_address) org_addr, slot, comp_code));
break;
case ADDRESS_DEVICE:
slot = TRB_GET_SLOT(trb.command); // slots are 1 based
if (BX_XHCI_THIS hub.slots[slot].enabled == 1) {
get_dwords((bx_phy_address) trb.parameter, (Bit32u*)buffer, (CONTEXT_SIZE + (CONTEXT_SIZE * 2)) >> 2);
bsr = ((trb.command & (1<<9)) == (1<<9));
DEV_MEM_READ_PHYSICAL((bx_phy_address) trb.parameter, 4, (Bit8u*)&tmpval1);
DEV_MEM_READ_PHYSICAL((bx_phy_address) trb.parameter + 4, 4, (Bit8u*)&tmpval2);
if ((tmpval1 == 0x00) && (tmpval2 == 0x03)) {
// Use temporary slot and ep context incase there is an error we don't modify the main contexts
copy_slot_from_buffer(&slot_context, &buffer[CONTEXT_SIZE]);
copy_ep_from_buffer(&ep_context, &buffer[CONTEXT_SIZE + CONTEXT_SIZE]);
// Only check for valid context on the first issued ADDRESS_DEVICE command
if ((BX_XHCI_THIS hub.slots[slot].sent_address == 1) ||
((validate_slot_context(&slot_context) && validate_ep_context(&ep_context, -1, 1)))) {
if (bsr == 1) { // BSR flag set
if (BX_XHCI_THIS hub.slots[slot].slot_context.slot_state == SLOT_STATE_DISABLED_ENABLED) {
slot_context.slot_state = SLOT_STATE_DEFAULT;
slot_context.device_address = 0;
ep_context.ep_state = EP_STATE_RUNNING;
comp_code = TRB_SUCCESS;
} else
comp_code = CONTEXT_STATE_ERROR;
} else { // BSR flag is clear
if (BX_XHCI_THIS hub.slots[slot].slot_context.slot_state <= SLOT_STATE_DEFAULT) {
int port_num = slot_context.rh_port_num - 1; // slot:port_num is 1 based
new_addr = create_unique_address(slot);
if (send_set_address(new_addr, port_num) == 0) {
slot_context.slot_state = SLOT_STATE_ADDRESSED;
slot_context.device_address = new_addr;
ep_context.ep_state = EP_STATE_RUNNING;
comp_code = TRB_SUCCESS;
} else
comp_code = TRANSACTION_ERROR;
} else
comp_code = CONTEXT_STATE_ERROR;
}
} else // validate contexts
comp_code = CONTEXT_STATE_ERROR;
} else
comp_code = CONTEXT_STATE_ERROR; // A0 and A1 not set correctly
} else
comp_code = SLOT_NOT_ENABLED;
// if successful, copy to the buffer allocated for this slot
if (comp_code == TRB_SUCCESS) {
memcpy(&BX_XHCI_THIS hub.slots[slot].slot_context, &slot_context, sizeof(struct SLOT_CONTEXT));
memcpy(&BX_XHCI_THIS hub.slots[slot].ep_context[1].ep_context, &ep_context, sizeof(struct EP_CONTEXT));
// initialize our internal enqueue pointer
BX_XHCI_THIS hub.slots[slot].ep_context[1].enqueue_pointer = ep_context.tr_dequeue_pointer;
BX_XHCI_THIS hub.slots[slot].ep_context[1].rcs = ep_context.dcs;
update_slot_context(slot);
update_ep_context(slot, 1);
// mark that we have done this once before
BX_XHCI_THIS hub.slots[slot].sent_address = 1;
}
write_event_TRB(0, org_addr, TRB_SET_COMP_CODE(comp_code), TRB_SET_SLOT(slot) | TRB_SET_TYPE(COMMAND_COMPLETION), 1);
//BX_INFO(("ADDRESS_DEVICE TRB: 0x" FMT_ADDRX64 " 0x%08X 0x%08X", trb.parameter, trb.status, trb.command));
BX_DEBUG(("0x" FORMATADDRESS ": Command Ring: SetAddress TRB (bsr = %i) (addr = %i) (slot = %i) (returning %i)",
(bx_phy_address) org_addr, bsr, new_addr, slot, comp_code));
break;
case EVALUATE_CONTEXT: {
slot = TRB_GET_SLOT(trb.command); // slots are 1 based
if (BX_XHCI_THIS hub.slots[slot].enabled == 1) {
get_dwords((bx_phy_address) trb.parameter, (Bit32u*)buffer, (CONTEXT_SIZE + (CONTEXT_SIZE * 32)) >> 2);
DEV_MEM_READ_PHYSICAL((bx_phy_address) trb.parameter + 4, 4, (Bit8u*)&a_flags);
// only the Slot context and EP1 (control EP) contexts are evaluated. Section 6.2.3.3
// If the slot is not addressed or configured, then return error
// XHCI specs 1.0, page 102 says DEFAULT or higher, while page 321 states higher than DEFAULT!!!
// (specs 1.1 removed the DEFAULT word and require it to be greater than the DEFAULT state)
#if ((VERSION_MAJOR == 1) && (VERSION_MINOR >= 0x10))
if (BX_XHCI_THIS hub.slots[slot].slot_context.slot_state > SLOT_STATE_DEFAULT) {
#else
if (BX_XHCI_THIS hub.slots[slot].slot_context.slot_state >= SLOT_STATE_DEFAULT) {
#endif
comp_code = TRB_SUCCESS; // assume good completion
if (a_flags & (1<<0)) {
copy_slot_from_buffer(&slot_context, &buffer[CONTEXT_SIZE]);
if (!validate_slot_context(&slot_context))
comp_code = PARAMETER_ERROR;
}
if (a_flags & (1<<1)) {
copy_ep_from_buffer(&ep_context, &buffer[CONTEXT_SIZE + CONTEXT_SIZE]);
if (!validate_ep_context(&ep_context, BX_XHCI_THIS hub.slots[slot].slot_context.speed, 1))
comp_code = PARAMETER_ERROR;
}
} else
comp_code = CONTEXT_STATE_ERROR;
// if all were good, go ahead and update our contexts
if (comp_code == TRB_SUCCESS) {
for (i=0; i<32; i++) {
if (a_flags & (1<<i)) {
if (i == 0) {
copy_slot_from_buffer(&slot_context, &buffer[CONTEXT_SIZE]);
// only the Interrupter Target and Max Exit Latency are updated by this command
BX_XHCI_THIS hub.slots[slot].slot_context.int_target = slot_context.int_target;
BX_XHCI_THIS hub.slots[slot].slot_context.max_exit_latency = slot_context.max_exit_latency;
// update the DCBAAP slot
update_slot_context(slot);
} else { // is an ep
// See section 4.8.2 for what fields will be updated
copy_ep_from_buffer(&ep_context, &buffer[CONTEXT_SIZE + (i * CONTEXT_SIZE)]);
// All types get these four updated
BX_XHCI_THIS hub.slots[slot].ep_context[i].ep_context.ep_type = ep_context.ep_type;
//BX_XHCI_THIS hub.slots[slot].ep_context[i].ep_context.tr_dequeue_pointer = ep_context.tr_dequeue_pointer;
BX_XHCI_THIS hub.slots[slot].ep_context[i].ep_context.dcs = 1;
// the Control EP0 endpoint's MPS field can not be modified by system software.
// it should remain the mps for the device attached.
if (i == 1) {
int port_num = BX_XHCI_THIS hub.slots[slot].slot_context.rh_port_num - 1;
switch (BX_XHCI_THIS hub.usb_port[port_num].device->get_speed()) {
case USB_SPEED_LOW:
BX_XHCI_THIS hub.slots[slot].ep_context[i].ep_context.max_packet_size = 8;
break;
case USB_SPEED_FULL:
case USB_SPEED_HIGH:
BX_XHCI_THIS hub.slots[slot].ep_context[i].ep_context.max_packet_size = 64;
break;
case USB_SPEED_SUPER:
BX_XHCI_THIS hub.slots[slot].ep_context[i].ep_context.max_packet_size = 512;
break;
}
} else
BX_XHCI_THIS hub.slots[slot].ep_context[i].ep_context.max_packet_size = ep_context.max_packet_size;
switch (ep_context.ep_type) {
case 4: // control
BX_XHCI_THIS hub.slots[slot].ep_context[i].ep_context.cerr = ep_context.cerr;
break;
case 2: // Bulk out
case 6: // Bulk in
BX_XHCI_THIS hub.slots[slot].ep_context[i].ep_context.max_burst_size = ep_context.max_burst_size;
BX_XHCI_THIS hub.slots[slot].ep_context[i].ep_context.cerr = ep_context.cerr;
BX_XHCI_THIS hub.slots[slot].ep_context[i].ep_context.max_pstreams = 0; // we don't support streams yet
break;
default: // ISO or interrupt
BX_XHCI_THIS hub.slots[slot].ep_context[i].ep_context.max_burst_size = ep_context.max_burst_size;
BX_XHCI_THIS hub.slots[slot].ep_context[i].ep_context.mult = ep_context.mult;
BX_XHCI_THIS hub.slots[slot].ep_context[i].ep_context.cerr = 3;
BX_XHCI_THIS hub.slots[slot].ep_context[i].ep_context.max_pstreams = 0; // we don't support streams yet
break;
}
// update our internal enqueue pointer
//BX_XHCI_THIS hub.slots[slot].ep_context[i].enqueue_pointer = ep_context.tr_dequeue_pointer;
BX_XHCI_THIS hub.slots[slot].ep_context[i].rcs = ep_context.dcs;
// update the DCBAAP slot's ep
update_ep_context(slot, i);
}
}
}
}
} else
comp_code = SLOT_NOT_ENABLED;
write_event_TRB(0, org_addr, TRB_SET_COMP_CODE(comp_code), TRB_SET_SLOT(slot) | TRB_SET_TYPE(COMMAND_COMPLETION), 1);
BX_DEBUG(("0x" FORMATADDRESS ": Command Ring: Evaluate TRB (slot = %i) (a_flags = 0x%08X) (returning %i)",
(bx_phy_address) org_addr, slot, a_flags, comp_code));
}
break;
case CONFIG_EP: {
slot = TRB_GET_SLOT(trb.command); // slots are 1 based
bool dc = TRB_DC(trb.command);
if (BX_XHCI_THIS hub.slots[slot].enabled) {
get_dwords((bx_phy_address) trb.parameter, (Bit32u*)buffer, (CONTEXT_SIZE + (CONTEXT_SIZE * 32)) >> 2);
DEV_MEM_READ_PHYSICAL((bx_phy_address) trb.parameter, 4, (Bit8u*)&d_flags);
DEV_MEM_READ_PHYSICAL((bx_phy_address) trb.parameter + 4, 4, (Bit8u*)&a_flags);
copy_slot_from_buffer(&slot_context, &buffer[CONTEXT_SIZE]); // so we get entry_count
if (BX_XHCI_THIS hub.slots[slot].slot_context.slot_state == SLOT_STATE_CONFIGURED) {
for (i=2; i<32; i++) {
if (dc || (d_flags & (1<<i))) {
BX_XHCI_THIS hub.slots[slot].ep_context[i].ep_context.ep_state = EP_STATE_DISABLED;
// TODO: Bandwidth stuff
update_ep_context(slot, i);
}
}
if (dc) {
BX_XHCI_THIS hub.slots[slot].slot_context.entries = 1;
update_slot_context(slot);
}
}
if (!dc && (BX_XHCI_THIS hub.slots[slot].slot_context.slot_state >= SLOT_STATE_ADDRESSED)) {
comp_code = TRB_SUCCESS; // assume good completion
// Check all the input context entries with an a_flag == 1
for (i=2; i<32; i++) {
if (a_flags & (1<<i)) {
copy_ep_from_buffer(&ep_context, &buffer[CONTEXT_SIZE + (CONTEXT_SIZE * i)]);
if ((i > (int) slot_context.entries) ||
!validate_ep_context(&ep_context, BX_XHCI_THIS hub.slots[slot].slot_context.speed, i)) {
comp_code = PARAMETER_ERROR;
break; // no need to check the rest
}
}
}
// if all were good, go ahead and update our contexts
if (comp_code == TRB_SUCCESS) {
for (i=2; i<32; i++) {
if (d_flags & (1<<i)) {
BX_XHCI_THIS hub.slots[slot].ep_context[i].ep_context.ep_state = EP_STATE_DISABLED;
// TODO: Bandwidth stuff
update_ep_context(slot, i);
}
if (a_flags & (1<<i)) {
copy_ep_from_buffer(&BX_XHCI_THIS hub.slots[slot].ep_context[i].ep_context, &buffer[CONTEXT_SIZE + (i * CONTEXT_SIZE)]);
BX_XHCI_THIS hub.slots[slot].ep_context[i].ep_context.ep_state = EP_STATE_RUNNING;
// initialize our internal enqueue pointer
BX_XHCI_THIS hub.slots[slot].ep_context[i].enqueue_pointer =
BX_XHCI_THIS hub.slots[slot].ep_context[i].ep_context.tr_dequeue_pointer;
BX_XHCI_THIS hub.slots[slot].ep_context[i].rcs = BX_XHCI_THIS hub.slots[slot].ep_context[i].ep_context.dcs;
BX_XHCI_THIS hub.slots[slot].ep_context[i].edtla = 0;
// TODO: Bandwidth stuff
update_ep_context(slot, i);
}
}
// if all EP's are in the disabled state, then set to slot_state = ADDRESSED, else slot_state = CONFIGURED
BX_XHCI_THIS hub.slots[slot].slot_context.slot_state = SLOT_STATE_ADDRESSED; // assume all disabled
for (i=2; i<32; i++) {
if (BX_XHCI_THIS hub.slots[slot].ep_context[i].ep_context.ep_state > EP_STATE_DISABLED) {
BX_XHCI_THIS hub.slots[slot].slot_context.slot_state = SLOT_STATE_CONFIGURED;
break;
}
}
BX_XHCI_THIS hub.slots[slot].slot_context.entries = slot_context.entries; ///////
update_slot_context(slot);
}
} else
comp_code = CONTEXT_STATE_ERROR;
} else
comp_code = SLOT_NOT_ENABLED;
// TODO: Page 101 (change to context entries)
write_event_TRB(0, org_addr, TRB_SET_COMP_CODE(comp_code), TRB_SET_SLOT(slot) | TRB_SET_TYPE(COMMAND_COMPLETION), 1);
BX_DEBUG(("0x" FORMATADDRESS ": Command Ring: Found Config_EP TRB (slot = %i) (returning %i)",
(bx_phy_address) org_addr, slot, comp_code));
}
break;
case SET_TR_DEQUEUE:
slot = TRB_GET_SLOT(trb.command); // slots are 1 based
ep = TRB_GET_EP(trb.command);
if (BX_XHCI_THIS hub.slots[slot].enabled == 1) {
if ((BX_XHCI_THIS hub.slots[slot].ep_context[ep].ep_context.ep_state == EP_STATE_STOPPED) ||
(BX_XHCI_THIS hub.slots[slot].ep_context[ep].ep_context.ep_state == EP_STATE_ERROR)) {
BX_XHCI_THIS hub.slots[slot].ep_context[ep].ep_context.tr_dequeue_pointer =
BX_XHCI_THIS hub.slots[slot].ep_context[ep].enqueue_pointer = (trb.parameter & (Bit64u) ~0xF);
BX_XHCI_THIS hub.slots[slot].ep_context[ep].ep_context.dcs =
BX_XHCI_THIS hub.slots[slot].ep_context[ep].rcs = (bool) (trb.parameter & 1);
BX_XHCI_THIS hub.slots[slot].ep_context[ep].edtla = 0;
update_ep_context(slot, ep);
comp_code = TRB_SUCCESS;
} else
comp_code = CONTEXT_STATE_ERROR;
} else
comp_code = SLOT_NOT_ENABLED;
// if the state is in error, move it to stopped
if (BX_XHCI_THIS hub.slots[slot].ep_context[ep].ep_context.ep_state == EP_STATE_ERROR)
BX_XHCI_THIS hub.slots[slot].ep_context[ep].ep_context.ep_state = EP_STATE_STOPPED;
write_event_TRB(0, org_addr, TRB_SET_COMP_CODE(comp_code), TRB_SET_SLOT(slot) | TRB_SET_TYPE(COMMAND_COMPLETION), 1);
BX_DEBUG(("0x" FORMATADDRESS ": Command Ring: Found Set_tr_Dequeue TRB (slot = %i) (ep = %i) (returning %i)",
(bx_phy_address) org_addr, slot, ep, comp_code));
if (comp_code == TRB_SUCCESS)
BX_INFO((" New address: 0x" FORMATADDRESS " state = %i", (bx_phy_address) BX_XHCI_THIS hub.slots[slot].ep_context[ep].enqueue_pointer,
BX_XHCI_THIS hub.slots[slot].ep_context[ep].ep_context.dcs));
break;
// case RESET_EP:
// May only be sent to EP's in the HALTED state. (page 105)
// BX_XHCI_THIS hub.slots[slot].ep_context[ep].ep_context.ep_state == EP_STATE_STOPPED;
// break;
case STOP_EP:
slot = TRB_GET_SLOT(trb.command); // slots are 1 based
ep = TRB_GET_EP(trb.command);
// A Stop Endpoint Command received while an endpoint is in the Error state shall have no effect and shall
// generate a Command Completion Event with the Completion Code set to Context State Error.
if (BX_XHCI_THIS hub.slots[slot].ep_context[ep].ep_context.ep_state == EP_STATE_ERROR)
comp_code = CONTEXT_STATE_ERROR;
else {
BX_XHCI_THIS hub.slots[slot].ep_context[ep].ep_context.ep_state = EP_STATE_STOPPED;
update_ep_context(slot, ep);
comp_code = TRB_SUCCESS;
}
write_event_TRB(0, org_addr, TRB_SET_COMP_CODE(comp_code), TRB_SET_SLOT(slot) | TRB_SET_TYPE(COMMAND_COMPLETION), 1);
BX_DEBUG(("0x" FORMATADDRESS ": Command Ring: Found Stop EP TRB (slot = %i) (ep = %i) (sp = %i) (returning 1)",
(bx_phy_address) org_addr, slot, ep, ((trb.command & (1<<23)) == (1<<23))));
break;
// Get Port Bandwidth (only available on version 0.96 or 0.95.)
case GET_PORT_BAND:
{
unsigned hub_id = TRB_GET_SLOT(trb.command);
unsigned band_speed = ((trb.command & (0x0F << 16)) >> 16) - 1;
if (hub_id == 0) { // root hub
if (band_speed < 4) {
DEV_MEM_WRITE_PHYSICAL_DMA((bx_phy_address) trb.parameter, 1 + USB_XHCI_PORTS, port_band_width[band_speed]);
comp_code = TRB_SUCCESS;
} else {
comp_code = TRB_ERROR;
BX_ERROR(("Get Port Bandwidth with unknown speed of %i", band_speed + 1));
}
} else {
#if SEC_DOMAIN_BAND
// TODO: External HUB support
comp_code = TRB_ERROR;
BX_ERROR(("Get Secondary Port Bandwidth not implemented yet."));
#else
write_event_TRB(0, org_addr, TRB_SET_COMP_CODE(TRB_ERROR), TRB_SET_SLOT(0) | TRB_SET_TYPE(COMMAND_COMPLETION), 1);
#endif
}
write_event_TRB(0, org_addr, TRB_SET_COMP_CODE(comp_code), TRB_SET_SLOT(0) | TRB_SET_TYPE(COMMAND_COMPLETION), 1);
BX_DEBUG(("0x" FORMATADDRESS ": Command Ring: GetPortBandwidth TRB (speed = %i) (hub_id = %i) (returning %i)",
(bx_phy_address) org_addr, band_speed, hub_id, comp_code));
}
break;
case RESET_DEVICE:
slot = TRB_GET_SLOT(trb.command); // slots are 1 based
if ((BX_XHCI_THIS hub.slots[slot].slot_context.slot_state == SLOT_STATE_ADDRESSED) ||
(BX_XHCI_THIS hub.slots[slot].slot_context.slot_state == SLOT_STATE_CONFIGURED)) {
BX_XHCI_THIS hub.slots[slot].slot_context.slot_state = SLOT_STATE_DEFAULT;
BX_XHCI_THIS hub.slots[slot].slot_context.entries = 1;
BX_XHCI_THIS hub.slots[slot].slot_context.device_address = 0;
update_slot_context(slot);
for (i=2; i<32; i++) {
BX_XHCI_THIS hub.slots[slot].ep_context[i].ep_context.ep_state = EP_STATE_DISABLED;
update_ep_context(slot, i);
}
comp_code = TRB_SUCCESS;
} else {
comp_code = CONTEXT_STATE_ERROR;
}
write_event_TRB(0, org_addr, TRB_SET_COMP_CODE(comp_code), TRB_SET_SLOT(slot) | TRB_SET_TYPE(COMMAND_COMPLETION), 1);
BX_DEBUG(("0x" FORMATADDRESS ": Command Ring: Found Reset Device TRB (slot = %i) (returning %i)",
(bx_phy_address) org_addr, slot, comp_code));
break;
// No Op (command ring) TRB (Sect: 6.4.3.1)
case NO_OP_CMD:
slot = TRB_GET_SLOT(trb.command); // slots are 1 based
ep = TRB_GET_EP(trb.command);
BX_DEBUG(("0x" FORMATADDRESS ": Command Ring (slot = %i) (ep = %i): Found No Op TRB",
(bx_phy_address) org_addr, slot, ep));
write_event_TRB(0, org_addr, TRB_SET_COMP_CODE(TRB_SUCCESS), TRB_SET_SLOT(0) | TRB_SET_TYPE(COMMAND_COMPLETION), 1);
break;
// unknown TRB type
default:
BX_ERROR(("0x" FORMATADDRESS ": Command Ring: Unknown TRB found.", (bx_phy_address) org_addr));
BX_ERROR(("Unknown trb type found: %i(dec) (0x" FMT_ADDRX64 " 0x%08X 0x%08X)", TRB_GET_TYPE(trb.command),
trb.parameter, trb.status, trb.command));
slot = TRB_GET_SLOT(trb.command); // slots are 1 based
write_event_TRB(0, 0x0, TRB_SET_COMP_CODE(TRB_ERROR), TRB_SET_SLOT(slot) | TRB_SET_TYPE(COMMAND_COMPLETION), 1);
}
// advance the Dequeue pointer and continue;
BX_XHCI_THIS hub.ring_members.command_ring.dq_pointer += 16;
read_TRB((bx_phy_address) BX_XHCI_THIS hub.ring_members.command_ring.dq_pointer, &trb);
}
}
void bx_usb_xhci_c::init_event_ring(const unsigned interrupter)
{
const Bit64u addr = BX_XHCI_THIS hub.runtime_regs.interrupter[interrupter].erstba.erstabadd;
Bit32u val32;
Bit64u val64;
BX_XHCI_THIS hub.ring_members.event_rings[interrupter].rcs = 1;
BX_XHCI_THIS hub.ring_members.event_rings[interrupter].count = 0;
DEV_MEM_READ_PHYSICAL_DMA((bx_phy_address) addr, sizeof(BX_XHCI_THIS hub.ring_members.event_rings[interrupter].entrys),
(Bit8u *) BX_XHCI_THIS hub.ring_members.event_rings[interrupter].entrys);
BX_XHCI_THIS hub.ring_members.event_rings[interrupter].cur_trb =
BX_XHCI_THIS hub.ring_members.event_rings[interrupter].entrys[0].addr;
BX_XHCI_THIS hub.ring_members.event_rings[interrupter].trb_count =
BX_XHCI_THIS hub.ring_members.event_rings[interrupter].entrys[0].size;
// dump the event segment table
BX_DEBUG(("Interrupter %02i: Event Ring Table (at 0x" FMT_ADDRX64 ") has %i entries:", interrupter,
addr, BX_XHCI_THIS hub.runtime_regs.interrupter[interrupter].erstsz.erstabsize));
for (int i=0; i<BX_XHCI_THIS hub.runtime_regs.interrupter[interrupter].erstsz.erstabsize; i++) {
DEV_MEM_READ_PHYSICAL((bx_phy_address) addr + (i * 16), 8, (Bit8u*)&val64);
DEV_MEM_READ_PHYSICAL((bx_phy_address) addr + (i * 16) + 8, 4, (Bit8u*)&val32);
BX_DEBUG((" %02i: address = 0x" FMT_ADDRX64 " Count = %i", i, val64, val32 & 0x0000FFFF));
}
}
void bx_usb_xhci_c::write_event_TRB(const unsigned interrupter, const Bit64u parameter, const Bit32u status,
const Bit32u command, const bool fire_int)
{
// write the TRB
write_TRB((bx_phy_address) BX_XHCI_THIS hub.ring_members.event_rings[interrupter].cur_trb, parameter, status,
command | (Bit32u)BX_XHCI_THIS hub.ring_members.event_rings[interrupter].rcs); // set the cycle bit
// calculate position for next event TRB
BX_XHCI_THIS hub.ring_members.event_rings[interrupter].cur_trb += 16;
BX_XHCI_THIS hub.ring_members.event_rings[interrupter].trb_count--;
if (BX_XHCI_THIS hub.ring_members.event_rings[interrupter].trb_count == 0) {
BX_XHCI_THIS hub.ring_members.event_rings[interrupter].count++;
if (BX_XHCI_THIS hub.ring_members.event_rings[interrupter].count ==
BX_XHCI_THIS hub.runtime_regs.interrupter[interrupter].erstsz.erstabsize) {
BX_XHCI_THIS hub.ring_members.event_rings[interrupter].rcs ^= 1;
BX_XHCI_THIS hub.ring_members.event_rings[interrupter].count = 0;
}
BX_XHCI_THIS hub.ring_members.event_rings[interrupter].cur_trb =
BX_XHCI_THIS hub.ring_members.event_rings[interrupter].entrys[BX_XHCI_THIS hub.ring_members.event_rings[interrupter].count].addr;
BX_XHCI_THIS hub.ring_members.event_rings[interrupter].trb_count =
BX_XHCI_THIS hub.ring_members.event_rings[interrupter].entrys[BX_XHCI_THIS hub.ring_members.event_rings[interrupter].count].size;
}
// if caller wants us to fire and interrupt, do so
if (fire_int) {
BX_XHCI_THIS hub.runtime_regs.interrupter[interrupter].iman.ip = 1;
BX_XHCI_THIS hub.runtime_regs.interrupter[interrupter].erdp.ehb = 1; // set event handler busy
BX_XHCI_THIS hub.op_regs.HcStatus.eint = 1;
update_irq(interrupter);
}
}
void bx_usb_xhci_c::read_TRB(bx_phy_address addr, struct TRB *trb)
{
DEV_MEM_READ_PHYSICAL(addr, 8, (Bit8u*)&trb->parameter);
DEV_MEM_READ_PHYSICAL(addr + 8, 4, (Bit8u*)&trb->status);
DEV_MEM_READ_PHYSICAL(addr + 12, 4, (Bit8u*)&trb->command);
}
void bx_usb_xhci_c::write_TRB(bx_phy_address addr, const Bit64u parameter, const Bit32u status, const Bit32u command)
{
DEV_MEM_WRITE_PHYSICAL(addr , 8, (Bit8u*)&parameter);
DEV_MEM_WRITE_PHYSICAL(addr + 8, 4, (Bit8u*)&status);
DEV_MEM_WRITE_PHYSICAL(addr + 12, 4, (Bit8u*)&command);
}
void bx_usb_xhci_c::update_slot_context(const int slot)
{
Bit32u buffer[16];
memset(buffer, 0, 64);
copy_slot_to_buffer(buffer, slot);
Bit64u slot_addr = (BX_XHCI_THIS hub.op_regs.HcDCBAAP.dcbaap + (slot * sizeof(Bit64u)));
DEV_MEM_READ_PHYSICAL((bx_phy_address) slot_addr, sizeof(Bit64u), (Bit8u *) &slot_addr);
put_dwords((bx_phy_address) slot_addr, buffer, CONTEXT_SIZE >> 2);
}
void bx_usb_xhci_c::update_ep_context(const int slot, const int ep)
{
Bit32u buffer[16];
memset(buffer, 0, 64);
copy_ep_to_buffer(buffer, slot, ep);
Bit64u slot_addr = (BX_XHCI_THIS hub.op_regs.HcDCBAAP.dcbaap + (slot * sizeof(Bit64u)));
DEV_MEM_READ_PHYSICAL((bx_phy_address) slot_addr, sizeof(Bit64u), (Bit8u *) &slot_addr);
put_dwords((bx_phy_address) (slot_addr + (ep * CONTEXT_SIZE)), buffer, CONTEXT_SIZE >> 2);
}
void bx_usb_xhci_c::dump_slot_context(const Bit32u *context, const int slot)
{
BX_INFO((" -=-=-=-=-=-=-=- Slot Context -=-=-=-=-=-=-=-"));
BX_INFO((" Context Entries: %i (%i)", (context[0] & (0x1F<<27)) >> 27, BX_XHCI_THIS hub.slots[slot].slot_context.entries));
BX_INFO((" Hub: %i (%i)", (context[0] & (1 <<26)) >> 26, BX_XHCI_THIS hub.slots[slot].slot_context.hub));
BX_INFO((" MTT: %i (%i)", (context[0] & (1 <<25)) >> 25, BX_XHCI_THIS hub.slots[slot].slot_context.mtt));
BX_INFO((" ReservedZ: %02X", (context[0] & (1 <<24)) >> 24));
BX_INFO((" Speed: %i (%i)", (context[0] & (0x0F<<20)) >> 20, BX_XHCI_THIS hub.slots[slot].slot_context.speed));
BX_INFO((" Route String: %05X (%05X)", (context[0] & 0xFFFFF) >> 0, BX_XHCI_THIS hub.slots[slot].slot_context.route_string));
BX_INFO((" Num Ports: %i (%i)", (context[1] & (0xFF<<24)) >> 24, BX_XHCI_THIS hub.slots[slot].slot_context.num_ports));
BX_INFO((" RH Port Num: %i (%i)", (context[1] & (0xFF<<16)) >> 16, BX_XHCI_THIS hub.slots[slot].slot_context.rh_port_num));
BX_INFO(("Max Exit Latency: %i (%i)", (context[1] & 0xFFFF) >> 0, BX_XHCI_THIS hub.slots[slot].slot_context.max_exit_latency));
BX_INFO((" Int Target: %i (%i)", (context[2] & (0x3F<<22)) >> 22, BX_XHCI_THIS hub.slots[slot].slot_context.int_target));
BX_INFO((" ReservedZ: %02X", (context[2] & (0x0F<<18)) >> 18));
BX_INFO((" TTT: %i (%i)", (context[2] & (0x03<<16)) >> 16, BX_XHCI_THIS hub.slots[slot].slot_context.ttt));
BX_INFO((" TT Port Num: %i (%i)", (context[2] & (0xFF<< 8)) >> 8, BX_XHCI_THIS hub.slots[slot].slot_context.tt_port_num));
BX_INFO((" TT Hub Slot: %i (%i)", (context[2] & 0xFF) >> 0, BX_XHCI_THIS hub.slots[slot].slot_context.tt_hub_slot_id));
BX_INFO((" Slot State: %i (%i)", (context[3] & (0x1F<<27)) >> 27, BX_XHCI_THIS hub.slots[slot].slot_context.slot_state));
BX_INFO((" ReservedZ: %06X", (context[3] & (0x7FFFF<<8)) >> 8));
BX_INFO((" Dev Address: %i (%i)", (context[3] & 0xFF) >> 0, BX_XHCI_THIS hub.slots[slot].slot_context.device_address));
BX_INFO((" ReservedZ: %08X", context[4]));
BX_INFO((" ReservedZ: %08X", context[5]));
BX_INFO((" ReservedZ: %08X", context[6]));
BX_INFO((" ReservedZ: %08X", context[7]));
#if (CONTEXT_SIZE == 64)
BX_INFO((" ReservedZ: %08x", context[8]));
BX_INFO((" ReservedZ: %08x", context[9]));
BX_INFO((" ReservedZ: %08x", context[10]));
BX_INFO((" ReservedZ: %08x", context[11]));
BX_INFO((" ReservedZ: %08x", context[12]));
BX_INFO((" ReservedZ: %08x", context[13]));
BX_INFO((" ReservedZ: %08x", context[14]));
BX_INFO((" ReservedZ: %08x", context[15]));
#endif
}
void bx_usb_xhci_c::dump_ep_context(const Bit32u *context, const int slot, const int ep)
{
BX_INFO((" -=-=-=-=-=-=-=-=- EP Context -=-=-=-=-=-=-=-"));
BX_INFO((" ReservedZ: %02x", (context[0] & (0xFF<<24)) >> 24));
BX_INFO((" Interval: %i (%i)", (context[0] & (0x0F<<16)) >> 16, BX_XHCI_THIS hub.slots[slot].ep_context[ep].ep_context.interval));
BX_INFO((" LSA: %i (%i)", (context[0] & (1 <<15)) >> 15, BX_XHCI_THIS hub.slots[slot].ep_context[ep].ep_context.lsa));
BX_INFO((" MaxPStreams: %i (%i)", (context[0] & (0x1F<<10)) >> 10, BX_XHCI_THIS hub.slots[slot].ep_context[ep].ep_context.max_pstreams));
BX_INFO((" Mult: %i (%i)", (context[0] & (0x03<< 8)) >> 8, BX_XHCI_THIS hub.slots[slot].ep_context[ep].ep_context.mult));
BX_INFO((" ReservedZ: %02x", (context[0] & (0x1F<< 3)) >> 3));
BX_INFO((" EP State: %i (%i)", (context[0] & 0x7) >> 0, BX_XHCI_THIS hub.slots[slot].ep_context[ep].ep_context.ep_state));
BX_INFO((" Max Packet Size: %i (%i)", (context[1] & (0xFFFF<<16)) >> 16, BX_XHCI_THIS hub.slots[slot].ep_context[ep].ep_context.max_packet_size));
BX_INFO((" Max Burst Size: %i (%i)", (context[1] & (0xFF<< 8)) >> 8, BX_XHCI_THIS hub.slots[slot].ep_context[ep].ep_context.max_burst_size));
BX_INFO((" HID: %i (%i)", (context[1] & (1 << 7)) >> 7, BX_XHCI_THIS hub.slots[slot].ep_context[ep].ep_context.hid));
BX_INFO((" ReservedZ: %01x", (context[1] & (1 << 6)) >> 6));
BX_INFO((" EP Type: %i (%i)", (context[1] & (0x07<< 3)) >> 3, BX_XHCI_THIS hub.slots[slot].ep_context[ep].ep_context.ep_type));
BX_INFO((" CErr: %i (%i)", (context[1] & (0x03<< 1)) >> 1, BX_XHCI_THIS hub.slots[slot].ep_context[ep].ep_context.cerr));
BX_INFO((" ReservedZ: %01x", (context[1] & (1 << 0)) >> 0));
BX_INFO((" TR Dequeue Ptr: " FMT_ADDRX64 " (" FMT_ADDRX64 ")", (*(Bit64u *) &context[2] & (Bit64u) ~0x0F), BX_XHCI_THIS hub.slots[slot].ep_context[ep].ep_context.tr_dequeue_pointer));
BX_INFO((" ReservedZ: %01x", (context[2] & (0x07 << 1)) >> 1));
BX_INFO((" DCS: %i (%i)", (context[2] & (1 << 0)) >> 0, BX_XHCI_THIS hub.slots[slot].ep_context[ep].ep_context.dcs));
BX_INFO(("Avg ESIT Payload: %i (%i)", (context[4] & (0xFFFF<<16)) >> 16, BX_XHCI_THIS hub.slots[slot].ep_context[ep].ep_context.max_esit_payload));
BX_INFO((" Avg TRB Length: %i (%i)", (context[4] & (0xFFFF<< 0)) >> 0, BX_XHCI_THIS hub.slots[slot].ep_context[ep].ep_context.average_trb_len));
BX_INFO((" ReservedZ: %08x", context[5]));
BX_INFO((" ReservedZ: %08x", context[6]));
BX_INFO((" ReservedZ: %08x", context[7]));
#if (CONTEXT_SIZE == 64)
BX_INFO((" ReservedZ: %08x", context[8]));
BX_INFO((" ReservedZ: %08x", context[9]));
BX_INFO((" ReservedZ: %08x", context[10]));
BX_INFO((" ReservedZ: %08x", context[11]));
BX_INFO((" ReservedZ: %08x", context[12]));
BX_INFO((" ReservedZ: %08x", context[13]));
BX_INFO((" ReservedZ: %08x", context[14]));
BX_INFO((" ReservedZ: %08x", context[15]));
#endif
}
void bx_usb_xhci_c::copy_slot_from_buffer(struct SLOT_CONTEXT *slot_context, const Bit8u *buffer)
{
Bit32u *buffer32 = (Bit32u*)buffer;
slot_context->entries = (buffer32[0] >> 27);
slot_context->hub = (buffer32[0] & (1<<26)) ? 1 : 0;
slot_context->mtt = (buffer32[0] & (1<<25)) ? 1 : 0;
slot_context->speed = (buffer32[0] & (0x0F<<20)) >> 20;
slot_context->route_string = (buffer32[0] & 0x000FFFFF);
slot_context->num_ports = (buffer32[1] >> 24);
slot_context->rh_port_num = (buffer32[1] & (0xFF<<16)) >> 16;
slot_context->max_exit_latency = (buffer32[1] & 0xFFFF);
slot_context->int_target = (buffer32[2] >> 22);
slot_context->ttt = (buffer32[2] & (0x3<<16)) >> 16;
slot_context->tt_port_num = (buffer32[2] & (0xFF<<8)) >> 8;
slot_context->tt_hub_slot_id = (buffer32[2] & 0xFF);
slot_context->slot_state = (buffer32[3] >> 27);
slot_context->device_address = (buffer32[3] & 0xFF);
}
void bx_usb_xhci_c::copy_ep_from_buffer(struct EP_CONTEXT *ep_context, const Bit8u *buffer)
{
Bit32u *buffer32 = (Bit32u*)buffer;
ep_context->interval = (buffer32[0] & (0xFF<<16)) >> 16;
ep_context->lsa = (buffer32[0] & (1<<15)) ? 1 : 0;
ep_context->max_pstreams = (buffer32[0] & (0x1F<<10)) >> 10;
ep_context->mult = (buffer32[0] & (0x03<<8)) >> 8;
ep_context->ep_state = (buffer32[0] & 0x07);
ep_context->max_packet_size = (buffer32[1] >> 16);
ep_context->max_burst_size = (buffer32[1] & (0xFF<<8)) >> 8;
ep_context->hid = (buffer32[1] & (1<<7)) ? 1 : 0;
ep_context->ep_type = (buffer32[1] & (0x07<<3)) >> 3;
ep_context->cerr = (buffer32[1] & (0x03<<1)) >> 1;
ep_context->tr_dequeue_pointer = (buffer32[2] & (Bit64u) ~0xF) | ((Bit64u)buffer32[3] << 32);
ep_context->dcs = (buffer32[2] & (1<<0));
ep_context->max_esit_payload = (buffer32[4] >> 16);
ep_context->average_trb_len = (buffer32[4] & 0xFFFF);
}
void bx_usb_xhci_c::copy_slot_to_buffer(Bit32u *buffer32, const int slot)
{
buffer32[0] = (BX_XHCI_THIS hub.slots[slot].slot_context.entries << 27) |
(BX_XHCI_THIS hub.slots[slot].slot_context.hub << 26) |
(BX_XHCI_THIS hub.slots[slot].slot_context.mtt << 25) |
(BX_XHCI_THIS hub.slots[slot].slot_context.speed << 20) |
BX_XHCI_THIS hub.slots[slot].slot_context.route_string;
buffer32[1] = (BX_XHCI_THIS hub.slots[slot].slot_context.num_ports << 24) |
(BX_XHCI_THIS hub.slots[slot].slot_context.rh_port_num << 16) |
BX_XHCI_THIS hub.slots[slot].slot_context.max_exit_latency;
buffer32[2] = (BX_XHCI_THIS hub.slots[slot].slot_context.int_target << 22) |
(BX_XHCI_THIS hub.slots[slot].slot_context.ttt << 16) |
(BX_XHCI_THIS hub.slots[slot].slot_context.tt_port_num << 8) |
BX_XHCI_THIS hub.slots[slot].slot_context.tt_hub_slot_id;
buffer32[3] = (BX_XHCI_THIS hub.slots[slot].slot_context.slot_state << 27) |
BX_XHCI_THIS hub.slots[slot].slot_context.device_address;
}
void bx_usb_xhci_c::copy_ep_to_buffer(Bit32u *buffer32, const int slot, const int ep)
{
buffer32[0] = (BX_XHCI_THIS hub.slots[slot].ep_context[ep].ep_context.interval << 16) |
(BX_XHCI_THIS hub.slots[slot].ep_context[ep].ep_context.lsa << 15) |
(BX_XHCI_THIS hub.slots[slot].ep_context[ep].ep_context.max_pstreams << 10) |
(BX_XHCI_THIS hub.slots[slot].ep_context[ep].ep_context.mult << 8) |
BX_XHCI_THIS hub.slots[slot].ep_context[ep].ep_context.ep_state;
buffer32[1] = (BX_XHCI_THIS hub.slots[slot].ep_context[ep].ep_context.max_packet_size << 16) |
(BX_XHCI_THIS hub.slots[slot].ep_context[ep].ep_context.max_burst_size << 8) |
(BX_XHCI_THIS hub.slots[slot].ep_context[ep].ep_context.hid << 7) |
(BX_XHCI_THIS hub.slots[slot].ep_context[ep].ep_context.ep_type << 3) |
(BX_XHCI_THIS hub.slots[slot].ep_context[ep].ep_context.cerr << 1);
buffer32[2] = ((Bit32u)BX_XHCI_THIS hub.slots[slot].ep_context[ep].ep_context.tr_dequeue_pointer) |
(Bit32u)BX_XHCI_THIS hub.slots[slot].ep_context[ep].ep_context.dcs;
buffer32[3] = (Bit32u)(BX_XHCI_THIS hub.slots[slot].ep_context[ep].ep_context.tr_dequeue_pointer >> 32);
buffer32[4] = (BX_XHCI_THIS hub.slots[slot].ep_context[ep].ep_context.max_esit_payload << 16) |
BX_XHCI_THIS hub.slots[slot].ep_context[ep].ep_context.average_trb_len;
}
bool bx_usb_xhci_c::validate_slot_context(const struct SLOT_CONTEXT *slot_context)
{
// specs:6.2.2.3: Only checks the Interrupter Target and Max Latency fields for validity
// See also 4.6.7
// only the Interrupter Target and Max Exit Latency are evaluated
BX_DEBUG((" slot_context->int_target = %i, slot_context->max_exit_latency = %i", slot_context->int_target, slot_context->max_exit_latency));
return 1;
}
// xHCI, section 6.2.3.2
// The controller can return a valid EP Context even though the MPS is less than
// the specified amount. Win7 uses this to detect the actual EP0's max packet size...
// For example, the descriptor might return 64, but the device really only handles
// 8-byte control transfers.
bool bx_usb_xhci_c::validate_ep_context(const struct EP_CONTEXT *ep_context, int speed, int ep_num)
{
// Only the Max_packet Size is evaluated (for an evaluate ep command) ???
BX_DEBUG((" ep_num = %i, speed = %i, ep_context->max_packet_size = %i", ep_num, speed, ep_context->max_packet_size));
// We need to make sure we don't exceed the value given in the
// device descriptor mps field.
// TODO: get the mps from the device's descriptor.
// for now, we just use the speed indicator passed to us...
unsigned int mps = 0;
switch (speed) {
case SPEED_LOW:
mps = 8;
break;
case SPEED_FULL:
case SPEED_HI:
mps = 64;
break;
case SPEED_SUPER:
mps = 512;
}
// if speed == -1, don't check the speed
if ((ep_num == 1) && (speed != -1)) {
// if not a mutliple of 8, return invalid
if ((ep_context->max_packet_size & 7) > 0)
return 0;
// if not at least 8, return invalid
if (ep_context->max_packet_size < 8)
return 0;
switch (speed) {
case SPEED_LOW:
// low-speed EP0 can only be a size of 8 bytes
return (ep_context->max_packet_size == 8);
case SPEED_FULL:
case SPEED_HI:
case SPEED_SUPER:
// full- and high-speed EP0 can be 8, 16, 32, 40, 48, 56, or 64
// super-speed EP0 can be 8, 16, 32, 40, 48, 56, 64, ..., 512
return ((ep_context->max_packet_size >= 8) &&
(ep_context->max_packet_size <= mps));
}
}
return 1;
}
// The Specs say that the address is only unique to the RH Port Number
// For now, we simply return the slot number (+1);
int bx_usb_xhci_c::create_unique_address(const int slot)
{
return (slot + 1); // Windows may need the first one to be 2 (though it shouldn't know the difference for xHCI)
}
int bx_usb_xhci_c::send_set_address(const int addr, const int port_num)
{
int ret;
USBPacket packet;
static Bit8u setup_address[8] = { 0, 0x05, 0, 0, 0, 0, 0 };
setup_address[2] = addr & 0xff;
setup_address[3] = addr >> 8;
packet.pid = USB_TOKEN_SETUP;
packet.devep = 0;
packet.devaddr = 0; // default address
packet.len = 8;
packet.data = setup_address;
packet.complete_cb = NULL;
packet.complete_dev = BX_XHCI_THIS_PTR;
ret = BX_XHCI_THIS broadcast_packet(&packet, port_num);
if (ret == 0) {
packet.pid = USB_TOKEN_IN;
packet.len = 0;
ret = BX_XHCI_THIS broadcast_packet(&packet, port_num);
}
return ret;
}
int bx_usb_xhci_c::broadcast_packet(USBPacket *p, const int port)
{
int ret = USB_RET_NODEV;
if (BX_XHCI_THIS hub.usb_port[port].device != NULL)
ret = BX_XHCI_THIS hub.usb_port[port].device->handle_packet(p);
return ret;
}
void bx_usb_xhci_c::xhci_timer_handler(void *this_ptr)
{
bx_usb_xhci_c *class_ptr = (bx_usb_xhci_c *) this_ptr;
class_ptr->xhci_timer();
}
void bx_usb_xhci_c::xhci_timer(void)
{
int slot, ep;
if (BX_XHCI_THIS hub.op_regs.HcStatus.hch)
return;
for (slot=1; slot<MAX_SLOTS; slot++) {
if (BX_XHCI_THIS hub.slots[slot].enabled) {
for (ep=1; ep<32; ep++) {
if (BX_XHCI_THIS hub.slots[slot].ep_context[ep].retry) {
if (--BX_XHCI_THIS hub.slots[slot].ep_context[ep].retry_counter <= 0) {
BX_XHCI_THIS process_transfer_ring(slot, ep);
}
}
}
}
}
}
void bx_usb_xhci_c::runtime_config_handler(void *this_ptr)
{
bx_usb_xhci_c *class_ptr = (bx_usb_xhci_c *) this_ptr;
class_ptr->runtime_config();
}
void bx_usb_xhci_c::runtime_config(void)
{
int i;
char pname[6];
int type = -1;
for (i = 0; i < BX_N_USB_XHCI_PORTS; i++) {
// device change support
if ((BX_XHCI_THIS device_change & (1 << i)) != 0) {
if (!BX_XHCI_THIS hub.usb_port[i].portsc.ccs) {
BX_INFO(("USB port #%d: device connect", i+1));
sprintf(pname, "port%d", i + 1);
init_device(i, (bx_list_c*)SIM->get_param(pname, SIM->get_param(BXPN_USB_XHCI)));
} else {
BX_INFO(("USB port #%d: device disconnect", i+1));
if (BX_XHCI_THIS hub.usb_port[i].device != NULL) {
type = BX_XHCI_THIS hub.usb_port[i].device->get_type();
}
usb_set_connect_status(i, type, 0);
}
BX_XHCI_THIS device_change &= ~(1 << i);
}
// forward to connected device
if (BX_XHCI_THIS hub.usb_port[i].device != NULL) {
BX_XHCI_THIS hub.usb_port[i].device->runtime_config();
}
}
}
// pci configuration space write callback handler
void bx_usb_xhci_c::pci_write_handler(Bit8u address, Bit32u value, unsigned io_len)
{
if (((address >= 0x14) && (address <= 0x34)))
return;
BX_DEBUG_PCI_WRITE(address, value, io_len);
for (unsigned i=0; i<io_len; i++) {
Bit8u value8 = (value >> (i*8)) & 0xFF;
// Bit8u oldval = BX_XHCI_THIS pci_conf[address+i];
switch (address+i) {
case 0x04:
value8 &= 0x06; // (bit 0 is read only for this card) (we don't allow port IO)
BX_XHCI_THIS pci_conf[address+i] = value8;
break;
case 0x3d: //
case 0x3e: //
case 0x3f: //
case 0x05: // disallowing write to command hi-byte
case 0x06: // disallowing write to status lo-byte (is that expected?)
break;
case 0x54:
if ((((value8 & 0x03) == 0x03) && ((BX_XHCI_THIS pci_conf[address+i] & 0x03) == 0x00)) &&
(BX_XHCI_THIS hub.op_regs.HcCommand.rs || !BX_XHCI_THIS hub.op_regs.HcStatus.hch))
BX_ERROR(("Power Transition from D0 to D3 with Run bit set and/or Halt bit clear"));
BX_XHCI_THIS pci_conf[address+i] = value8;
break;
case 0x55:
BX_XHCI_THIS pci_conf[address+i] = value8;
if (value8 & 0x80) // if we write a one to bit 7 (15 of word register at 0x54), clear bit 7.
BX_XHCI_THIS pci_conf[address+i] &= 0x7F;
break;
default:
BX_XHCI_THIS pci_conf[address+i] = value8;
}
}
}
void bx_usb_xhci_c::usb_set_connect_status(Bit8u port, int type, bool connected)
{
const bool ccs_org = BX_XHCI_THIS hub.usb_port[port].portsc.ccs;
const bool ped_org = BX_XHCI_THIS hub.usb_port[port].portsc.ped;
usb_device_c *device = BX_XHCI_THIS hub.usb_port[port].device;
if (device != NULL) {
if (device->get_type() == type) {
if (connected) {
if ((device->get_speed() == USB_SPEED_SUPER) &&
!BX_XHCI_THIS hub.usb_port[port].is_usb3) {
BX_PANIC(("Super-speed device not supported on USB2 port."));
usb_set_connect_status(port, type, 0);
return;
}
if (BX_XHCI_THIS hub.usb_port[port].is_usb3) {
if (!device->set_speed(USB_SPEED_SUPER)) {
BX_PANIC(("Only super-speed devices supported on USB3 port."));
usb_set_connect_status(port, type, 0);
return;
}
}
switch (device->get_speed()) {
case USB_SPEED_LOW:
BX_XHCI_THIS hub.usb_port[port].portsc.speed = 2;
break;
case USB_SPEED_FULL:
BX_XHCI_THIS hub.usb_port[port].portsc.speed = 1;
break;
case USB_SPEED_HIGH:
BX_XHCI_THIS hub.usb_port[port].portsc.speed = 3;
break;
case USB_SPEED_SUPER:
BX_XHCI_THIS hub.usb_port[port].portsc.speed = 4;
break;
default:
BX_PANIC(("USB device returned invalid speed value"));
usb_set_connect_status(port, type, 0);
return;
}
BX_XHCI_THIS hub.usb_port[port].portsc.ccs = 1;
if (!device->get_connected()) {
if (!device->init()) {
usb_set_connect_status(port, type, 0);
BX_ERROR(("port #%d: connect failed", port+1));
return;
} else {
BX_INFO(("port #%d: connect: %s", port+1, device->get_info()));
}
}
device->set_event_handler(BX_XHCI_THIS_PTR, xhci_event_handler, port);
} else { // not connected
BX_XHCI_THIS hub.usb_port[port].portsc.ccs = 0;
BX_XHCI_THIS hub.usb_port[port].portsc.ped = 0;
BX_XHCI_THIS hub.usb_port[port].portsc.speed = 0;
remove_device(port);
}
}
if (ccs_org != BX_XHCI_THIS hub.usb_port[port].portsc.ccs)
BX_XHCI_THIS hub.usb_port[port].portsc.csc = 1;
if (ped_org != BX_XHCI_THIS hub.usb_port[port].portsc.ped)
BX_XHCI_THIS hub.usb_port[port].portsc.pec = 1;
// we changed the value of the port, so show it
if (!BX_XHCI_THIS hub.op_regs.HcStatus.hch) {
BX_INFO(("Port #%d Status Change Event.", port + 1));
write_event_TRB(0, ((port + 1) << 24), TRB_SET_COMP_CODE(1), TRB_SET_TYPE(PORT_STATUS_CHANGE), 1);
}
}
}
// USB runtime parameter handler
const char *bx_usb_xhci_c::usb_param_handler(bx_param_string_c *param, int set,
const char *oldval, const char *val, int maxlen)
{
int portnum;
if (set) {
portnum = atoi((param->get_parent())->get_name()+4) - 1;
bool empty = ((strlen(val) == 0) || (!strcmp(val, "none")));
if ((portnum >= 0) && (portnum < USB_XHCI_PORTS)) {
if (empty && BX_XHCI_THIS hub.usb_port[portnum].portsc.ccs) {
BX_XHCI_THIS device_change |= (1 << portnum);
} else if (!empty && !BX_XHCI_THIS hub.usb_port[portnum].portsc.ccs) {
BX_XHCI_THIS device_change |= (1 << portnum);
}
} else {
BX_PANIC(("usb_param_handler called with unexpected parameter '%s'", param->get_name()));
}
}
return val;
}
void bx_usb_xhci_c::dump_xhci_core(const int slots, const int eps)
{
bx_phy_address addr = BX_XHCI_THIS pci_bar[0].addr;
Bit32u dword;
Bit64u qword, slot_addr;
int p, i;
Bit8u buffer[4096];
// dump the caps registers
BX_INFO((" CAPLENGTH: 0x%02X", BX_XHCI_THIS hub.cap_regs.HcCapLength & 0xFF));
BX_INFO(("HC VERSION: %X.%02X", ((BX_XHCI_THIS hub.cap_regs.HcCapLength & 0xFF000000) >> 24),
((BX_XHCI_THIS hub.cap_regs.HcCapLength & 0x00FF0000) >> 16)));
BX_INFO(("HCSPARAMS1: 0x%08X", BX_XHCI_THIS hub.cap_regs.HcSParams1));
BX_INFO(("HCSPARAMS2: 0x%08X", BX_XHCI_THIS hub.cap_regs.HcSParams2));
BX_INFO(("HCSPARAMS3: 0x%08X", BX_XHCI_THIS hub.cap_regs.HcSParams3));
#if ((VERSION_MAJOR == 1) && (VERSION_MINOR >= 0x10))
BX_INFO(("HCCPARAMS1: 0x%08X", BX_XHCI_THIS hub.cap_regs.HcCParams1));
#else
BX_INFO(("HCCPARAMS: 0x%08X", BX_XHCI_THIS hub.cap_regs.HcCParams1));
#endif
BX_INFO((" DBOFF: 0x%08X", BX_XHCI_THIS hub.cap_regs.DBOFF));
BX_INFO((" RTSOFF: 0x%08X", BX_XHCI_THIS hub.cap_regs.RTSOFF));
#if ((VERSION_MAJOR == 1) && (VERSION_MINOR >= 0x10))
BX_INFO(("HCCPARAMS2: 0x%08X", BX_XHCI_THIS hub.cap_regs.HcCParams2));
#endif
// dump the operational registers
BX_XHCI_THIS read_handler(addr + 0x20, 4, &dword, NULL);
BX_INFO((" USB_COMMAND: 0x%08X", dword));
BX_XHCI_THIS read_handler(addr + 0x24, 4, &dword, NULL);
BX_INFO((" USB_STATUS: 0x%08X", dword));
BX_XHCI_THIS read_handler(addr + 0x28, 4, &dword, NULL);
BX_INFO((" PAGE_SIZE: 0x%08X", dword));
BX_XHCI_THIS read_handler(addr + 0x34, 4, &dword, NULL);
BX_INFO((" DNCTRL: 0x%08X", dword));
BX_XHCI_THIS read_handler(addr + 0x38, 8, &qword, NULL);
BX_INFO((" CRCR: 0x" FMT_ADDRX64, qword));
BX_XHCI_THIS read_handler(addr + 0x50, 8, &qword, NULL);
BX_INFO((" DCBAAP: 0x" FMT_ADDRX64, qword));
BX_XHCI_THIS read_handler(addr + 0x58, 4, &dword, NULL);
BX_INFO((" CONFIG: 0x%08X", dword));
for (i=0, p=0; i<USB_XHCI_PORTS; i++, p+=16) {
BX_XHCI_THIS read_handler(addr + 0x420 + (p + 0), 4, &dword, NULL);
BX_INFO((" Port %i: 0x%08X", i, dword));
BX_XHCI_THIS read_handler(addr + 0x420 + (p + 4), 4, &dword, NULL);
BX_INFO((" 0x%08X", dword));
BX_XHCI_THIS read_handler(addr + 0x420 + (p + 8), 4, &dword, NULL);
BX_INFO((" 0x%08X", dword));
BX_XHCI_THIS read_handler(addr + 0x420 + (p + 12), 4, &dword, NULL);
BX_INFO((" 0x%08X", dword));
}
slot_addr = BX_XHCI_THIS hub.op_regs.HcDCBAAP.dcbaap;
DEV_MEM_READ_PHYSICAL((bx_phy_address) slot_addr, sizeof(Bit64u), (Bit8u *) &slot_addr);
BX_INFO((" SCRATCH PADS: 0x" FMT_ADDRX64, slot_addr));
for (i=1; i<slots+1; i++) {
slot_addr = (BX_XHCI_THIS hub.op_regs.HcDCBAAP.dcbaap + (i * sizeof(Bit64u)));
DEV_MEM_READ_PHYSICAL((bx_phy_address) slot_addr, sizeof(Bit64u), (Bit8u *) &slot_addr);
DEV_MEM_READ_PHYSICAL_DMA((bx_phy_address) slot_addr, 2048, buffer);
dump_slot_context((Bit32u *) &buffer[0], i);
for (p=1; p<eps+1; p++)
dump_ep_context((Bit32u *) &buffer[p * CONTEXT_SIZE], i, p);
}
}
#endif // BX_SUPPORT_PCI && BX_SUPPORT_USB_XHCI