2003-01-28 19:58:10 +03:00
|
|
|
/////////////////////////////////////////////////////////////////////////
|
2011-02-25 01:05:47 +03:00
|
|
|
// $Id$
|
2003-01-28 19:58:10 +03:00
|
|
|
/////////////////////////////////////////////////////////////////////////
|
|
|
|
//
|
2015-06-29 23:32:56 +03:00
|
|
|
// Copyright (C) 2009-2015 Benjamin D Lunt (fys [at] fysnet [dot] net)
|
2021-01-23 15:06:11 +03:00
|
|
|
// 2009-2021 The Bochs Project
|
2003-01-28 19:58:10 +03:00
|
|
|
//
|
|
|
|
// 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
|
2009-02-08 12:05:52 +03:00
|
|
|
// Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
|
2009-02-08 00:05:31 +03:00
|
|
|
/////////////////////////////////////////////////////////////////////////
|
2003-01-28 19:58:10 +03:00
|
|
|
|
2015-04-07 19:57:36 +03:00
|
|
|
// USB UHCI adapter
|
2012-11-12 22:56:07 +04:00
|
|
|
// PIIX3/PIIX4 function 2
|
2009-01-18 16:11:27 +03:00
|
|
|
|
2015-11-06 00:20:54 +03:00
|
|
|
// Notes by Ben Lunt (see uhci_core.cc)
|
2003-01-28 19:58:10 +03:00
|
|
|
|
|
|
|
// Define BX_PLUGGABLE in files that can be compiled into plugins. For
|
2008-01-27 01:24:03 +03:00
|
|
|
// platforms that require a special tag on exported symbols, BX_PLUGGABLE
|
2003-01-28 19:58:10 +03:00
|
|
|
// is used to know when we are exporting symbols and when we are importing.
|
|
|
|
#define BX_PLUGGABLE
|
|
|
|
|
2004-06-19 19:20:15 +04:00
|
|
|
#include "iodev.h"
|
2009-01-03 11:55:00 +03:00
|
|
|
|
2009-01-19 12:48:12 +03:00
|
|
|
#if BX_SUPPORT_PCI && BX_SUPPORT_USB_UHCI
|
2009-01-03 11:55:00 +03:00
|
|
|
|
|
|
|
#include "pci.h"
|
2009-01-18 16:11:27 +03:00
|
|
|
#include "usb_common.h"
|
2015-11-06 00:20:54 +03:00
|
|
|
#include "uhci_core.h"
|
2009-01-19 12:48:12 +03:00
|
|
|
#include "usb_uhci.h"
|
2003-01-28 19:58:10 +03:00
|
|
|
|
2009-01-19 12:48:12 +03:00
|
|
|
#define LOG_THIS theUSB_UHCI->
|
2003-01-28 19:58:10 +03:00
|
|
|
|
2009-01-19 12:48:12 +03:00
|
|
|
bx_usb_uhci_c* theUSB_UHCI = NULL;
|
2003-01-28 19:58:10 +03:00
|
|
|
|
2012-01-08 16:43:46 +04:00
|
|
|
// builtin configuration handling functions
|
|
|
|
|
|
|
|
Bit32s usb_uhci_options_parser(const char *context, int num_params, char *params[])
|
|
|
|
{
|
|
|
|
if (!strcmp(params[0], "usb_uhci")) {
|
|
|
|
bx_list_c *base = (bx_list_c*) SIM->get_param(BXPN_USB_UHCI);
|
|
|
|
for (int i = 1; i < num_params; i++) {
|
|
|
|
if (!strncmp(params[i], "enabled=", 8)) {
|
|
|
|
SIM->get_param_bool(BXPN_UHCI_ENABLED)->set(atol(¶ms[i][8]));
|
|
|
|
} else if (!strncmp(params[i], "port", 4)) {
|
2015-10-31 18:12:36 +03:00
|
|
|
if (SIM->parse_usb_port_params(context, 0, params[i], USB_UHCI_PORTS, base) < 0) {
|
2012-01-08 16:43:46 +04:00
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
} else if (!strncmp(params[i], "options", 7)) {
|
2015-10-31 18:12:36 +03:00
|
|
|
if (SIM->parse_usb_port_params(context, 1, params[i], USB_UHCI_PORTS, base) < 0) {
|
2012-01-08 16:43:46 +04:00
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
BX_ERROR(("%s: unknown parameter '%s' for usb_uhci ignored.", context, params[i]));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
BX_PANIC(("%s: unknown directive '%s'", context, params[0]));
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
Bit32s usb_uhci_options_save(FILE *fp)
|
|
|
|
{
|
|
|
|
bx_list_c *base = (bx_list_c*) SIM->get_param(BXPN_USB_UHCI);
|
2015-10-31 18:12:36 +03:00
|
|
|
SIM->write_usb_options(fp, USB_UHCI_PORTS, base);
|
2012-01-08 16:43:46 +04:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2021-01-23 15:06:11 +03:00
|
|
|
// device plugin entry point
|
2012-01-08 16:43:46 +04:00
|
|
|
|
2021-01-23 15:06:11 +03:00
|
|
|
PLUGIN_ENTRY_FOR_MODULE(usb_uhci)
|
2003-01-28 19:58:10 +03:00
|
|
|
{
|
2021-02-07 19:16:06 +03:00
|
|
|
if (mode == PLUGIN_INIT) {
|
2021-01-23 15:06:11 +03:00
|
|
|
theUSB_UHCI = new bx_usb_uhci_c();
|
|
|
|
BX_REGISTER_DEVICE_DEVMODEL(plugin, type, theUSB_UHCI, BX_PLUGIN_USB_UHCI);
|
|
|
|
// add new configuration parameter for the config interface
|
|
|
|
SIM->init_usb_options("UHCI", "uhci", USB_UHCI_PORTS);
|
|
|
|
// register add-on option for bochsrc and command line
|
|
|
|
SIM->register_addon_option("usb_uhci", usb_uhci_options_parser, usb_uhci_options_save);
|
2021-02-07 19:16:06 +03:00
|
|
|
} else if (mode == PLUGIN_FINI) {
|
2021-01-23 15:06:11 +03:00
|
|
|
SIM->unregister_addon_option("usb_uhci");
|
|
|
|
bx_list_c *menu = (bx_list_c*)SIM->get_param("ports.usb");
|
|
|
|
delete theUSB_UHCI;
|
|
|
|
menu->remove("uhci");
|
2021-02-07 19:16:06 +03:00
|
|
|
} else {
|
|
|
|
return (int)PLUGTYPE_OPTIONAL;
|
2021-01-23 15:06:11 +03:00
|
|
|
}
|
2003-01-28 19:58:10 +03:00
|
|
|
return 0; // Success
|
|
|
|
}
|
|
|
|
|
2012-01-08 16:43:46 +04:00
|
|
|
// the device object
|
|
|
|
|
2009-01-19 12:48:12 +03:00
|
|
|
bx_usb_uhci_c::bx_usb_uhci_c()
|
2003-01-28 19:58:10 +03:00
|
|
|
{
|
2012-01-08 16:43:46 +04:00
|
|
|
put("usb_uhci", "UHCI");
|
2015-11-06 00:20:54 +03:00
|
|
|
rt_conf_id = -1;
|
2003-01-28 19:58:10 +03:00
|
|
|
}
|
|
|
|
|
2009-01-19 12:48:12 +03:00
|
|
|
bx_usb_uhci_c::~bx_usb_uhci_c()
|
2003-01-28 19:58:10 +03:00
|
|
|
{
|
2011-01-16 15:46:48 +03:00
|
|
|
char pname[16];
|
2009-03-04 21:20:50 +03:00
|
|
|
|
2015-11-06 00:20:54 +03:00
|
|
|
SIM->unregister_runtime_config_handler(rt_conf_id);
|
2007-03-14 21:05:46 +03:00
|
|
|
|
2015-10-31 18:12:36 +03:00
|
|
|
for (int i=0; i<USB_UHCI_PORTS; i++) {
|
2011-01-16 15:46:48 +03:00
|
|
|
sprintf(pname, "port%d.device", i+1);
|
2009-03-04 21:20:50 +03:00
|
|
|
SIM->get_param_string(pname, SIM->get_param(BXPN_USB_UHCI))->set_handler(NULL);
|
2009-02-27 01:46:37 +03:00
|
|
|
remove_device(i);
|
2007-03-14 21:05:46 +03:00
|
|
|
}
|
2005-11-08 21:49:45 +03:00
|
|
|
|
2012-08-19 12:16:20 +04:00
|
|
|
SIM->get_bochs_root()->remove("usb_uhci");
|
2013-02-17 12:27:43 +04:00
|
|
|
bx_list_c *usb_rt = (bx_list_c*)SIM->get_param(BXPN_MENU_RUNTIME_USB);
|
|
|
|
usb_rt->remove("uhci");
|
2006-09-10 21:18:44 +04:00
|
|
|
BX_DEBUG(("Exit"));
|
2003-01-28 19:58:10 +03:00
|
|
|
}
|
|
|
|
|
2009-01-19 12:48:12 +03:00
|
|
|
void bx_usb_uhci_c::init(void)
|
2003-01-28 19:58:10 +03:00
|
|
|
{
|
2009-02-27 01:46:37 +03:00
|
|
|
unsigned i;
|
2009-03-04 21:20:50 +03:00
|
|
|
char pname[6];
|
2012-01-08 16:43:46 +04:00
|
|
|
bx_list_c *uhci, *port;
|
2013-11-26 01:07:39 +04:00
|
|
|
bx_param_string_c *device;
|
2015-11-06 00:20:54 +03:00
|
|
|
Bit8u devfunc;
|
2018-02-24 21:04:36 +03:00
|
|
|
Bit16u devid;
|
2003-01-28 19:58:10 +03:00
|
|
|
|
2012-01-08 16:43:46 +04:00
|
|
|
// Read in values from config interface
|
|
|
|
uhci = (bx_list_c*) SIM->get_param(BXPN_USB_UHCI);
|
|
|
|
// Check if the device is disabled or not configured
|
|
|
|
if (!SIM->get_param_bool("enabled", uhci)->get()) {
|
|
|
|
BX_INFO(("USB UHCI disabled"));
|
2012-07-06 21:19:32 +04:00
|
|
|
// mark unused plugin for removal
|
|
|
|
((bx_param_bool_c*)((bx_list_c*)SIM->get_param(BXPN_PLUGIN_CTRL))->get_by_name("usb_uhci"))->set(0);
|
2012-01-08 16:43:46 +04:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2012-11-11 12:11:17 +04:00
|
|
|
if (SIM->get_param_enum(BXPN_PCI_CHIPSET)->get() == BX_PCI_CHIPSET_I440FX) {
|
2018-02-24 21:04:36 +03:00
|
|
|
devfunc = BX_PCI_DEVICE(1, 2);
|
|
|
|
devid = 0x7020;
|
|
|
|
} else if (SIM->get_param_enum(BXPN_PCI_CHIPSET)->get() == BX_PCI_CHIPSET_I440BX) {
|
|
|
|
devfunc = BX_PCI_DEVICE(7, 2);
|
|
|
|
devid = 0x7112;
|
2012-11-11 12:11:17 +04:00
|
|
|
} else {
|
2015-11-06 00:20:54 +03:00
|
|
|
devfunc = 0x00;
|
2018-02-24 21:04:36 +03:00
|
|
|
devid = 0x7020;
|
2009-02-15 11:16:16 +03:00
|
|
|
}
|
2018-02-24 21:04:36 +03:00
|
|
|
BX_UHCI_THIS init_uhci(devfunc, devid, 0x00, BX_PCI_INTD);
|
2004-12-11 11:35:33 +03:00
|
|
|
|
2009-03-04 21:20:50 +03:00
|
|
|
bx_list_c *usb_rt = (bx_list_c*)SIM->get_param(BXPN_MENU_RUNTIME_USB);
|
2013-11-26 01:07:39 +04:00
|
|
|
bx_list_c *uhci_rt = new bx_list_c(usb_rt, "uhci", "UHCI Runtime Options");
|
|
|
|
uhci_rt->set_options(uhci_rt->SHOW_PARENT);
|
2015-10-31 18:12:36 +03:00
|
|
|
for (i=0; i<USB_UHCI_PORTS; i++) {
|
2009-03-04 21:20:50 +03:00
|
|
|
sprintf(pname, "port%d", i+1);
|
2011-01-16 15:46:48 +03:00
|
|
|
port = (bx_list_c*)SIM->get_param(pname, uhci);
|
2013-11-26 01:07:39 +04:00
|
|
|
uhci_rt->add(port);
|
2011-01-16 15:46:48 +03:00
|
|
|
device = (bx_param_string_c*)port->get_by_name("device");
|
|
|
|
device->set_handler(usb_param_handler);
|
2009-02-27 01:46:37 +03:00
|
|
|
}
|
|
|
|
|
2011-06-11 23:38:52 +04:00
|
|
|
// register handler for correct device connect handling after runtime config
|
2015-11-06 00:20:54 +03:00
|
|
|
BX_UHCI_THIS rt_conf_id = SIM->register_runtime_config_handler(BX_UHCI_THIS_PTR, runtime_config_handler);
|
|
|
|
BX_UHCI_THIS device_change = 0;
|
2009-04-06 13:30:26 +04:00
|
|
|
|
2009-01-19 12:48:12 +03:00
|
|
|
BX_INFO(("USB UHCI initialized"));
|
2003-01-28 19:58:10 +03:00
|
|
|
}
|
|
|
|
|
2009-01-19 12:48:12 +03:00
|
|
|
void bx_usb_uhci_c::reset(unsigned type)
|
2003-01-28 19:58:10 +03:00
|
|
|
{
|
2015-11-06 00:20:54 +03:00
|
|
|
unsigned i;
|
2009-03-04 21:20:50 +03:00
|
|
|
char pname[6];
|
2003-02-02 13:24:26 +03:00
|
|
|
|
2015-11-06 00:20:54 +03:00
|
|
|
BX_UHCI_THIS reset_uhci(type);
|
|
|
|
for (i=0; i<USB_UHCI_PORTS; i++) {
|
|
|
|
if (BX_UHCI_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_UHCI)));
|
2009-03-04 21:20:50 +03:00
|
|
|
}
|
2003-01-28 19:58:10 +03:00
|
|
|
}
|
2005-01-14 21:28:47 +03:00
|
|
|
}
|
|
|
|
|
2015-11-06 00:20:54 +03:00
|
|
|
void bx_usb_uhci_c::register_state()
|
2006-05-27 19:54:49 +04:00
|
|
|
{
|
2020-09-03 09:45:39 +03:00
|
|
|
BX_UHCI_THIS uhci_register_state(SIM->get_bochs_root());
|
2006-05-27 19:54:49 +04:00
|
|
|
}
|
|
|
|
|
2015-11-06 00:20:54 +03:00
|
|
|
void bx_usb_uhci_c::after_restore_state()
|
2006-05-27 19:54:49 +04:00
|
|
|
{
|
2015-11-06 00:20:54 +03:00
|
|
|
bx_uhci_core_c::after_restore_state();
|
2006-05-27 19:54:49 +04:00
|
|
|
}
|
|
|
|
|
2011-01-16 15:46:48 +03:00
|
|
|
void bx_usb_uhci_c::init_device(Bit8u port, bx_list_c *portconf)
|
2005-01-14 21:28:47 +03:00
|
|
|
{
|
2021-02-14 11:30:49 +03:00
|
|
|
int type;
|
2007-03-31 13:24:04 +04:00
|
|
|
char pname[BX_PATHNAME_LEN];
|
2011-01-16 15:46:48 +03:00
|
|
|
const char *devname = NULL;
|
2004-12-19 12:59:40 +03:00
|
|
|
|
2011-01-16 15:46:48 +03:00
|
|
|
devname = ((bx_param_string_c*)portconf->get_by_name("device"))->getptr();
|
|
|
|
if (devname == NULL) return;
|
2006-09-24 14:10:21 +04:00
|
|
|
if (!strlen(devname) || !strcmp(devname, "none")) return;
|
2004-12-19 12:59:40 +03:00
|
|
|
|
2009-02-15 17:06:55 +03:00
|
|
|
if (BX_UHCI_THIS hub.usb_port[port].device != NULL) {
|
|
|
|
BX_ERROR(("init_device(): port%d already in use", port+1));
|
|
|
|
return;
|
|
|
|
}
|
2010-12-15 00:20:37 +03:00
|
|
|
sprintf(pname, "usb_uhci.hub.port%d.device", port+1);
|
|
|
|
bx_list_c *sr_list = (bx_list_c*)SIM->get_param(pname, SIM->get_bochs_root());
|
2021-02-14 11:30:49 +03:00
|
|
|
type = DEV_usb_init_device(portconf, BX_UHCI_THIS_PTR, &BX_UHCI_THIS hub.usb_port[port].device);
|
2009-03-09 15:18:40 +03:00
|
|
|
if (BX_UHCI_THIS hub.usb_port[port].device != NULL) {
|
2015-11-08 21:54:30 +03:00
|
|
|
set_connect_status(port, type, 1);
|
2021-02-14 11:30:49 +03:00
|
|
|
BX_UHCI_THIS hub.usb_port[port].device->register_state(sr_list);
|
2005-01-14 21:28:47 +03:00
|
|
|
}
|
2004-12-11 11:35:33 +03:00
|
|
|
}
|
|
|
|
|
2009-02-27 01:46:37 +03:00
|
|
|
void bx_usb_uhci_c::remove_device(Bit8u port)
|
|
|
|
{
|
|
|
|
if (BX_UHCI_THIS hub.usb_port[port].device != NULL) {
|
2015-08-28 21:52:54 +03:00
|
|
|
delete BX_UHCI_THIS hub.usb_port[port].device;
|
|
|
|
BX_UHCI_THIS hub.usb_port[port].device = NULL;
|
2009-02-27 01:46:37 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-06-11 23:38:52 +04:00
|
|
|
void bx_usb_uhci_c::runtime_config_handler(void *this_ptr)
|
|
|
|
{
|
|
|
|
bx_usb_uhci_c *class_ptr = (bx_usb_uhci_c *) this_ptr;
|
|
|
|
class_ptr->runtime_config();
|
|
|
|
}
|
|
|
|
|
|
|
|
void bx_usb_uhci_c::runtime_config(void)
|
|
|
|
{
|
|
|
|
int i;
|
|
|
|
char pname[6];
|
2021-02-14 11:30:49 +03:00
|
|
|
int type = -1;
|
2011-06-11 23:38:52 +04:00
|
|
|
|
2015-10-31 18:12:36 +03:00
|
|
|
for (i = 0; i < USB_UHCI_PORTS; i++) {
|
2011-06-11 23:38:52 +04:00
|
|
|
// device change support
|
2015-11-06 00:20:54 +03:00
|
|
|
if ((BX_UHCI_THIS device_change & (1 << i)) != 0) {
|
2015-07-14 00:57:09 +03:00
|
|
|
if (!BX_UHCI_THIS hub.usb_port[i].status) {
|
|
|
|
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_UHCI)));
|
|
|
|
} else {
|
|
|
|
BX_INFO(("USB port #%d: device disconnect", i+1));
|
|
|
|
if (BX_UHCI_THIS hub.usb_port[i].device != NULL) {
|
|
|
|
type = BX_UHCI_THIS hub.usb_port[i].device->get_type();
|
|
|
|
}
|
2015-11-08 21:54:30 +03:00
|
|
|
set_connect_status(i, type, 0);
|
2015-11-06 00:20:54 +03:00
|
|
|
remove_device(i);
|
2015-07-14 00:57:09 +03:00
|
|
|
}
|
2015-11-06 00:20:54 +03:00
|
|
|
BX_UHCI_THIS device_change &= ~(1 << i);
|
2011-06-11 23:38:52 +04:00
|
|
|
}
|
|
|
|
// forward to connected device
|
|
|
|
if (BX_UHCI_THIS hub.usb_port[i].device != NULL) {
|
|
|
|
BX_UHCI_THIS hub.usb_port[i].device->runtime_config();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2007-03-14 21:05:46 +03:00
|
|
|
// USB runtime parameter handler
|
2009-01-19 12:48:12 +03:00
|
|
|
const char *bx_usb_uhci_c::usb_param_handler(bx_param_string_c *param, int set,
|
2021-02-01 23:39:33 +03:00
|
|
|
const char *oldval, const char *val, int maxlen)
|
2005-11-29 23:46:17 +03:00
|
|
|
{
|
2009-03-04 21:20:50 +03:00
|
|
|
int portnum;
|
2006-09-24 14:10:21 +04:00
|
|
|
|
2005-11-29 23:46:17 +03:00
|
|
|
if (set) {
|
2011-01-16 15:46:48 +03:00
|
|
|
portnum = atoi((param->get_parent())->get_name()+4) - 1;
|
2021-02-01 23:39:33 +03:00
|
|
|
bool empty = ((strlen(val) == 0) || (!strcmp(val, "none")));
|
2015-10-31 18:12:36 +03:00
|
|
|
if ((portnum >= 0) && (portnum < USB_UHCI_PORTS)) {
|
2009-03-05 22:12:23 +03:00
|
|
|
if (empty && BX_UHCI_THIS hub.usb_port[portnum].status) {
|
2015-11-06 00:20:54 +03:00
|
|
|
BX_UHCI_THIS device_change |= (1 << portnum);
|
2009-03-05 22:12:23 +03:00
|
|
|
} else if (!empty && !BX_UHCI_THIS hub.usb_port[portnum].status) {
|
2015-11-06 00:20:54 +03:00
|
|
|
BX_UHCI_THIS device_change |= (1 << portnum);
|
2006-09-24 14:10:21 +04:00
|
|
|
}
|
2006-03-01 20:14:36 +03:00
|
|
|
} else {
|
2009-03-04 21:20:50 +03:00
|
|
|
BX_PANIC(("usb_param_handler called with unexpected parameter '%s'", param->get_name()));
|
2005-11-29 23:46:17 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
return val;
|
|
|
|
}
|
|
|
|
|
2009-01-19 12:48:12 +03:00
|
|
|
#endif // BX_SUPPORT_PCI && BX_SUPPORT_USB_UHCI
|