ffc722f4e8
by Michele Giacomone Detailed description: -Observed issues Due to some limitations only dates between 1980 and 2038 can be used in a reliable way. Also, bochs incorrectly assumes a linear correspondence between the data returned by the <time.h> functions localtime() and mktime(), and isn't setting the latter properly. Bochs keeps its internal time value dependent to these functions after setup, assuming that their internal settings won't change on the go - which is not the case. In my OS, and in my timezone, this leads to incorrect startup values for 5 months each year and unreliable values if the simulation is kept going for a long time. (a feedback between localtime() and mktime() is created which keeps shifting back the time) Also, the RTC simulation is not realistic since the clock fixes itself across DST changes, without updating any DST related flag, a behavior that no guest OS expects. -Proposed fix This is implemented in such way that no bochs' previous behavior is changed, a part from the broken ones, with legacy in mind == the user can keep using bochs exactly as before knowing nothing of this patch +Make the internal s.timeval variable a Bit64s, so it can fit all values that the cmos can correctly represent, reported below: MIN setting -62167219200 => 0000/01/01 SAT 0:00:00 MAX BCD setting 253402300799 => 9999/12/31 FRI 23:59:59 MAX BIN setting 745690751999 => 25599/12/31 FRI 23:59:59 And then fix each reference to these so it can handle such values And make bochs correctly wrap around for under/overflows, so that only the most significant bits of the century are lost. +Do the same thing to the bochs time0 parameter, so all the above values can be chosen at startup (despite being now legal values, 1 and 2 will still be treated as "local" and "utc"). Note that normally only BCD settings are valid since bochs' CMOS defaults to such operating mode - the only way to use the binary range is by loading a cmos memory map. +Make the internal s.timeval variable independent from external factors. This means providing a small set of time handling functions, contained in "iodev/utctime.h", which must work in any environment in which bochs compiles, accessing no external resource. This also means that after startup, s.timeval will only be changed internally, and no call to the OS time functions will be made. +Make the internal s.timeval variable timezone independent, to have a linear correlation between its values and valid CMOS settings. To make it easier, s.timeval is gonna be treated as if the current timezone was UTC: so, - if the user selects UTC as time0, s.timeval will become current time(NULL) - if the user selects localtime, s.timeval will be computed as the value which will display the same broken down time as localtime(&now) - if the user inputs a time formatted string the proper s.timeval to displayed will be easily calculated, - if the user inputs a starting time value, s.timeval will be computed as the value which will display the same broken down time as localtime(&user_input) to ensure the same operation as before. A "tz=utc" is displayed when bochs prints out the current time value, to warn users about the difference in meaning between the internally kept time value and the value they can set through the "time0=" parameter. This might be changed to communicate instead the time value they can input to get the same setting, but performing such calculation (except for the startup time) suffers from all the mktime()/localtime() problems listed above so I did not do it. The range of "time0" is automatically adjusted so all users in all time zones can set any legal value despite "time0=" having a local meaning. A thorough explanation of what I did and why can be found in the "iodev/utctime.h" library header. --------- Co-authored-by: Stanislav Shwartsman <sshwarts@users.sourceforge.net> Co-authored-by: Volker Ruppert <Volker.Ruppert@t-online.de>
3685 lines
122 KiB
C++
3685 lines
122 KiB
C++
/////////////////////////////////////////////////////////////////////////
|
|
// $Id$
|
|
/////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// Copyright (C) 2002-2023 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
|
|
|
|
#include "bochs.h"
|
|
#include "bxversion.h"
|
|
#include "iodev/iodev.h"
|
|
#include "iodev/hdimage/hdimage.h"
|
|
#if BX_NETWORKING
|
|
#include "iodev/network/netmod.h"
|
|
#endif
|
|
#if BX_SUPPORT_SOUNDLOW
|
|
#include "iodev/sound/soundmod.h"
|
|
#endif
|
|
#if BX_SUPPORT_PCIUSB
|
|
#include "iodev/usb/usb_common.h"
|
|
#endif
|
|
#include "param_names.h"
|
|
#include <assert.h>
|
|
|
|
#ifdef HAVE_LOCALE_H
|
|
#include <locale.h>
|
|
#endif
|
|
|
|
#if defined(macintosh)
|
|
// Work around a bug in SDL 1.2.4 on MacOS X, which redefines getenv to
|
|
// SDL_getenv, but then neglects to provide SDL_getenv. It happens
|
|
// because we are defining -Dmacintosh.
|
|
#undef getenv
|
|
#endif
|
|
|
|
#ifdef WIN32
|
|
#define DIRECTORY_SEPARATOR "\\"
|
|
#else
|
|
#define DIRECTORY_SEPARATOR "/"
|
|
#endif
|
|
|
|
const char **config_interface_list;
|
|
const char **display_library_list;
|
|
const char **vga_extension_names;
|
|
const char **vga_extension_plugins;
|
|
const char **pcislot_dev_list;
|
|
int bochsrc_include_level = 0;
|
|
|
|
#define LOG_THIS genlog->
|
|
|
|
extern bx_debug_t bx_dbg;
|
|
|
|
static int parse_line_unformatted(const char *context, char *line);
|
|
static int parse_line_formatted(const char *context, int num_params, char *params[]);
|
|
static int parse_bochsrc(const char *rcfile);
|
|
static int get_floppy_type_from_image(const char *filename);
|
|
|
|
int get_floppy_devtype_from_type(int type)
|
|
{
|
|
switch (type) {
|
|
case BX_FLOPPY_2_88:
|
|
return BX_FDD_350ED;
|
|
case BX_FLOPPY_1_44:
|
|
return BX_FDD_350HD;
|
|
case BX_FLOPPY_720K:
|
|
return BX_FDD_350DD;
|
|
case BX_FLOPPY_1_2:
|
|
return BX_FDD_525HD;
|
|
// the remaining types return a 5 1/4" 360k drive
|
|
case BX_FLOPPY_360K:
|
|
default:
|
|
return BX_FDD_525DD;
|
|
}
|
|
}
|
|
|
|
static Bit64s bx_param_handler(bx_param_c *param, bool set, Bit64s val)
|
|
{
|
|
char pname[BX_PATHNAME_LEN];
|
|
Bit8u device, vga_ext_id;
|
|
Bit64s oldval;
|
|
|
|
bx_list_c *base = (bx_list_c*) param->get_parent();
|
|
base->get_param_path(pname, BX_PATHNAME_LEN);
|
|
if (!strncmp(pname, "ata.", 4)) {
|
|
if (!strcmp(base->get_name(), "master")) {
|
|
device = 0;
|
|
} else {
|
|
device = 1;
|
|
}
|
|
if (!strcmp(param->get_name(), "type")) {
|
|
if (set) {
|
|
switch (val) {
|
|
case BX_ATA_DEVICE_DISK:
|
|
((bx_param_filename_c*)SIM->get_param("path", base))->set_extension("img");
|
|
break;
|
|
case BX_ATA_DEVICE_CDROM:
|
|
((bx_param_filename_c*)SIM->get_param("path", base))->set_extension("iso");
|
|
break;
|
|
}
|
|
}
|
|
} else {
|
|
BX_PANIC(("bx_param_handler called with unknown parameter '%s.%s'", pname, param->get_name()));
|
|
return -1;
|
|
}
|
|
} else {
|
|
param->get_param_path(pname, BX_PATHNAME_LEN);
|
|
if (!strcmp(pname, BXPN_VGA_EXTENSION)) {
|
|
if (set) {
|
|
oldval = ((bx_param_enum_c*)param)->get();
|
|
if (val != oldval) {
|
|
vga_ext_id = (Bit8u)((bx_param_enum_c*)param)->get();
|
|
PLUG_unload_opt_plugin(vga_extension_plugins[vga_ext_id]);
|
|
PLUG_load_plugin_var(vga_extension_plugins[(Bit8u)val], PLUGTYPE_VGA);
|
|
}
|
|
}
|
|
} else if ((!strcmp(pname, BXPN_FLOPPYA_DEVTYPE)) ||
|
|
(!strcmp(pname, BXPN_FLOPPYB_DEVTYPE))) {
|
|
if ((set) && (val == BX_FDD_NONE)) {
|
|
SIM->get_param_enum("type", base)->set(BX_FLOPPY_NONE);
|
|
SIM->get_param_enum("status", base)->set(BX_EJECTED);
|
|
}
|
|
} else if ((!strcmp(pname, BXPN_FLOPPYA_TYPE)) ||
|
|
(!strcmp(pname, BXPN_FLOPPYB_TYPE))) {
|
|
if (set) {
|
|
if (val == BX_FLOPPY_AUTO) {
|
|
val = get_floppy_type_from_image(SIM->get_param_string("path", base)->getptr());
|
|
SIM->get_param_enum("type", base)->set(val);
|
|
} else if (!SIM->get_init_done() && (val != BX_FLOPPY_NONE)) {
|
|
device = get_floppy_devtype_from_type((int)val);
|
|
SIM->get_param_enum("devtype", base)->set(device);
|
|
}
|
|
}
|
|
} else {
|
|
BX_PANIC(("bx_param_handler called with unknown parameter '%s'", pname));
|
|
return -1;
|
|
}
|
|
}
|
|
return val;
|
|
}
|
|
|
|
const char *bx_param_string_handler(bx_param_string_c *param, bool set,
|
|
const char *oldval, const char *val, int maxlen)
|
|
{
|
|
char pname[BX_PATHNAME_LEN];
|
|
|
|
param->get_param_path(pname, BX_PATHNAME_LEN);
|
|
if (!strcmp(pname, BXPN_SCREENMODE)) {
|
|
if (set) {
|
|
BX_INFO(("Screen mode changed to %s", val));
|
|
}
|
|
} else if (!strcmp(pname, BXPN_USER_SHORTCUT)) {
|
|
if (set && (SIM->get_init_done())) {
|
|
if (!bx_gui->parse_user_shortcut(val)) {
|
|
val = oldval;
|
|
}
|
|
}
|
|
} else {
|
|
BX_PANIC(("bx_param_string_handler called with unknown parameter '%s'", pname));
|
|
}
|
|
return val;
|
|
}
|
|
|
|
#if BX_NETWORKING
|
|
void bx_init_std_nic_options(const char *name, bx_list_c *menu)
|
|
{
|
|
bx_param_enum_c *ethmod;
|
|
bx_param_bytestring_c *macaddr;
|
|
bx_param_filename_c *path, *bootrom;
|
|
char descr[120];
|
|
|
|
sprintf(descr, "MAC address of the %s device. Don't use an address of a machine on your net.", name);
|
|
macaddr = new bx_param_bytestring_c(menu,
|
|
"mac",
|
|
"MAC Address",
|
|
descr,
|
|
"", 6);
|
|
macaddr->set_initial_val("\xfe\xfd\xde\xad\xbe\xef");
|
|
macaddr->set_separator(':');
|
|
ethmod = new bx_param_enum_c(menu,
|
|
"ethmod",
|
|
"Ethernet module",
|
|
"Module used for the connection to the real net.",
|
|
bx_netmod_ctl.get_module_names(),
|
|
0,
|
|
0);
|
|
ethmod->set_by_name("null");
|
|
ethmod->set_ask_format("Choose ethernet module for the device [%s] ");
|
|
new bx_param_string_c(menu,
|
|
"ethdev",
|
|
"Ethernet device",
|
|
"Device used for the connection to the real net. This is only valid if an ethernet module other than 'null' is used.",
|
|
"xl0", BX_PATHNAME_LEN);
|
|
path = new bx_param_filename_c(menu,
|
|
"script",
|
|
"Device configuration script",
|
|
"Name of the script that is executed after Bochs initializes the network interface (optional).",
|
|
"none", BX_PATHNAME_LEN);
|
|
path->set_ask_format("Enter new script name, or 'none': [%s] ");
|
|
bootrom = new bx_param_filename_c(menu,
|
|
"bootrom",
|
|
"Boot ROM image",
|
|
"Pathname of network boot ROM image to load",
|
|
"", BX_PATHNAME_LEN);
|
|
bootrom->set_format("Name of boot ROM image: %s");
|
|
}
|
|
#endif
|
|
|
|
#if BX_SUPPORT_PCIUSB
|
|
void bx_init_usb_options(const char *usb_name, const char *pname, int maxports, int param0)
|
|
{
|
|
char group[16], name[8], descr[512], label[512];
|
|
bx_list_c *usb, *usbrt, *deplist, *deplist2;
|
|
|
|
bx_list_c *ports = (bx_list_c*)SIM->get_param("ports");
|
|
usb = (bx_list_c*)ports->get_by_name("usb");
|
|
if (usb == NULL) {
|
|
usb = new bx_list_c(ports, "usb", "USB Configuration");
|
|
usb->set_options(usb->USE_TAB_WINDOW | usb->SHOW_PARENT);
|
|
// prepare runtime options
|
|
bx_list_c *rtmenu = (bx_list_c*)SIM->get_param("menu.runtime");
|
|
usbrt = new bx_list_c(rtmenu, "usb", "USB options");
|
|
usbrt->set_runtime_param(1);
|
|
usbrt->set_options(usbrt->SHOW_PARENT | usbrt->USE_TAB_WINDOW);
|
|
}
|
|
sprintf(group, "USB %s", usb_name);
|
|
sprintf(label, "%s Configuration", usb_name);
|
|
bx_list_c *menu = new bx_list_c(usb, pname, label);
|
|
menu->set_options(menu->SHOW_PARENT);
|
|
sprintf(label, "Enable %s emulation", usb_name);
|
|
sprintf(descr, "Enables the %s emulation", usb_name);
|
|
bx_param_bool_c *enabled = new bx_param_bool_c(menu, "enabled", label, descr, 1);
|
|
|
|
// ehci companion type
|
|
static const char *ehci_comp_type[] = { "uhci", "ohci", NULL };
|
|
new bx_param_enum_c(menu,
|
|
"companion", "Companion Type",
|
|
"Select Companion type to emulate",
|
|
ehci_comp_type,
|
|
0, 0
|
|
);
|
|
|
|
// xhci host controller type and number of ports
|
|
static const char *xhci_model_names[] = { "uPD720202", "uPD720201", NULL };
|
|
new bx_param_enum_c(menu,
|
|
"model", "HC model",
|
|
"Select Host Controller to emulate",
|
|
xhci_model_names,
|
|
0, 0
|
|
);
|
|
new bx_param_num_c(menu,
|
|
"n_ports", "Number of ports",
|
|
"Set the number of ports for this controller",
|
|
-1, 10,
|
|
-1, 0 // -1 as a default so that we can tell if this parameter was given
|
|
);
|
|
|
|
deplist = new bx_list_c(NULL);
|
|
for (Bit8u i = 0; i < maxports; i++) {
|
|
sprintf(name, "port%u", i+1);
|
|
sprintf(label, "Port #%u Configuration", i+1);
|
|
sprintf(descr, "Device connected to %s port #%u and it's options", usb_name, i+1);
|
|
bx_list_c *port = new bx_list_c(menu, name, label);
|
|
port->set_options(port->SERIES_ASK | port->USE_BOX_TITLE);
|
|
sprintf(descr, "Device connected to %s port #%d", usb_name, i+1);
|
|
bx_param_enum_c *device = new bx_param_enum_c(port,
|
|
"device",
|
|
"Device",
|
|
descr,
|
|
bx_usbdev_ctl.get_device_names(),
|
|
0, 0);
|
|
sprintf(descr, "Options for device connected to %s port #%u", usb_name, i+1);
|
|
#if BX_SUPPORT_USB_XHCI
|
|
if (!strcmp(usb_name, "xHCI")) {
|
|
if (i < (param0 / 2))
|
|
strcpy(label, "Options: (Must be super-speed)");
|
|
else
|
|
strcpy(label, "Options: (Must NOT be super-speed)");
|
|
} else
|
|
#endif
|
|
strcpy(label, "Options");
|
|
bx_param_string_c *options = new bx_param_string_c(port,
|
|
"options",
|
|
label,
|
|
descr,
|
|
"", BX_PATHNAME_LEN);
|
|
bx_param_bool_c *overcurrent = new bx_param_bool_c(port,
|
|
"over_current",
|
|
"signal over-current",
|
|
"signal over-current", 0);
|
|
port->set_group(group);
|
|
deplist->add(port);
|
|
deplist->add(device);
|
|
deplist->add(overcurrent);
|
|
deplist2 = new bx_list_c(NULL);
|
|
deplist2->add(options);
|
|
device->set_dependent_list(deplist2, 1);
|
|
device->set_dependent_bitmap(0, 0);
|
|
}
|
|
enabled->set_dependent_list(deplist);
|
|
}
|
|
#endif
|
|
|
|
void bx_plugin_ctrl_init()
|
|
{
|
|
bx_list_c *base = (bx_list_c*) SIM->get_param(BXPN_PLUGIN_CTRL);
|
|
const char *name;
|
|
int count = PLUG_get_plugins_count(PLUGTYPE_OPTIONAL);
|
|
if (count == 0) {
|
|
BX_PANIC(("bx_plugin_ctrl_init() failure: no plugins found"));
|
|
}
|
|
for (int i = 0; i < count; i++) {
|
|
name = PLUG_get_plugin_name(PLUGTYPE_OPTIONAL, i);
|
|
new bx_param_bool_c(base, name, "", "", 0);
|
|
}
|
|
}
|
|
|
|
void bx_plugin_ctrl_reset(bool init_done)
|
|
{
|
|
bx_list_c *base = (bx_list_c*) SIM->get_param(BXPN_PLUGIN_CTRL);
|
|
if (init_done) {
|
|
for (int i = 0; i < base->get_size(); i++) {
|
|
((bx_param_bool_c*)base->get(i))->set(0);
|
|
}
|
|
SIM->opt_plugin_ctrl("*", 0);
|
|
}
|
|
// enable the default set of plugins to be loaded
|
|
SIM->get_param_bool("unmapped", base)->set(1);
|
|
SIM->get_param_bool("biosdev", base)->set(1);
|
|
SIM->get_param_bool("speaker", base)->set(1);
|
|
SIM->get_param_bool("extfpuirq", base)->set(1);
|
|
SIM->get_param_bool("parallel", base)->set(1);
|
|
SIM->get_param_bool("serial", base)->set(1);
|
|
#if BX_SUPPORT_GAMEPORT
|
|
SIM->get_param_bool("gameport", base)->set(1);
|
|
#endif
|
|
#if BX_SUPPORT_IODEBUG && BX_DEBUGGER
|
|
SIM->get_param_bool("iodebug", base)->set(1);
|
|
#endif
|
|
SIM->opt_plugin_ctrl("*", 1);
|
|
}
|
|
|
|
bool bx_opt_plugin_available(const char *plugname)
|
|
{
|
|
return (((bx_list_c*)SIM->get_param(BXPN_PLUGIN_CTRL))->get_by_name(plugname) != NULL);
|
|
}
|
|
|
|
void bx_init_config_interface_list()
|
|
{
|
|
Bit8u i, count = 0;
|
|
|
|
count = PLUG_get_plugins_count(PLUGTYPE_CI);
|
|
config_interface_list = (const char**) malloc((count + 1) * sizeof(char*));
|
|
for (i = 0; i < count; i++) {
|
|
config_interface_list[i] = PLUG_get_plugin_name(PLUGTYPE_CI, i);
|
|
}
|
|
config_interface_list[count] = NULL;
|
|
// move default config_intergface to the top of the list
|
|
if (strcmp(config_interface_list[0], BX_DEFAULT_CONFIG_INTERFACE)) {
|
|
for (i = 1; i < count; i++) {
|
|
if (!strcmp(config_interface_list[i], BX_DEFAULT_CONFIG_INTERFACE)) {
|
|
config_interface_list[i] = config_interface_list[0];
|
|
config_interface_list[0] = BX_DEFAULT_CONFIG_INTERFACE;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
void bx_init_displaylib_list()
|
|
{
|
|
Bit8u i, count = 0;
|
|
|
|
count = PLUG_get_plugins_count(PLUGTYPE_GUI);
|
|
display_library_list = (const char**) malloc((count + 1) * sizeof(char*));
|
|
for (i = 0; i < count; i++) {
|
|
display_library_list[i] = PLUG_get_plugin_name(PLUGTYPE_GUI, i);
|
|
}
|
|
display_library_list[count] = NULL;
|
|
// move default display library to the top of the list
|
|
if (strcmp(display_library_list[0], BX_DEFAULT_DISPLAY_LIBRARY)) {
|
|
for (i = 1; i < count; i++) {
|
|
if (!strcmp(display_library_list[i], BX_DEFAULT_DISPLAY_LIBRARY)) {
|
|
display_library_list[i] = display_library_list[0];
|
|
display_library_list[0] = BX_DEFAULT_DISPLAY_LIBRARY;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
void bx_init_vgaext_list()
|
|
{
|
|
Bit8u i, count = 0;
|
|
const char *plugname;
|
|
|
|
count = PLUG_get_plugins_count(PLUGTYPE_VGA);
|
|
vga_extension_names = (const char**) malloc((count + 2) * sizeof(char*));
|
|
vga_extension_plugins = (const char**) malloc((count + 1) * sizeof(char*));
|
|
vga_extension_names[0] = "none";
|
|
vga_extension_plugins[0] = "vga";
|
|
for (i = 0; i < count; i++) {
|
|
plugname = PLUG_get_plugin_name(PLUGTYPE_VGA, i);
|
|
vga_extension_plugins[i + 1] = plugname;
|
|
if (!strcmp(plugname, "vga")) {
|
|
vga_extension_names[i + 1] = "vbe";
|
|
} else if (!strcmp(plugname, "svga_cirrus")) {
|
|
vga_extension_names[i + 1] = plugname + 5;
|
|
} else {
|
|
vga_extension_names[i + 1] = plugname;
|
|
}
|
|
}
|
|
vga_extension_names[count + 1] = NULL;
|
|
}
|
|
|
|
void bx_init_pcidev_list()
|
|
{
|
|
Bit8u i, j, count, flags;
|
|
const Bit16u mask = PLUGTYPE_VGA | PLUGTYPE_OPTIONAL;
|
|
const char *plugname;
|
|
|
|
count = PLUG_get_plugins_count(mask);
|
|
pcislot_dev_list = (const char**) malloc((count + 2) * sizeof(char*));
|
|
pcislot_dev_list[0] = "none";
|
|
j = 1;
|
|
for (i = 0; i < count; i++) {
|
|
plugname = PLUG_get_plugin_name(mask, i);
|
|
flags = PLUG_get_plugin_flags(mask, i);
|
|
if ((flags & PLUGFLAG_PCI) != 0) {
|
|
if (!strcmp(plugname, "vga")) {
|
|
plugname = "pcivga";
|
|
} else if (!strcmp(plugname, "svga_cirrus")) {
|
|
plugname = "cirrus";
|
|
}
|
|
pcislot_dev_list[j++] = plugname;
|
|
}
|
|
}
|
|
pcislot_dev_list[j] = NULL;
|
|
}
|
|
|
|
void bx_init_options()
|
|
{
|
|
int i;
|
|
bx_list_c *menu;
|
|
bx_list_c *deplist;
|
|
bx_param_num_c *ioaddr, *ioaddr2, *irq;
|
|
bx_param_bool_c *enabled, *readonly;
|
|
bx_param_enum_c *mode, *type, *toggle, *status;
|
|
bx_param_filename_c *path;
|
|
char name[BX_PATHNAME_LEN], descr[512], label[512];
|
|
|
|
bx_param_c *root_param = SIM->get_param(".");
|
|
|
|
// general options subtree
|
|
menu = new bx_list_c(root_param, "general", "");
|
|
|
|
// config interface option, set in bochsrc or command line
|
|
bx_init_config_interface_list();
|
|
bx_param_enum_c *sel_config = new bx_param_enum_c(menu,
|
|
"config_interface", "Configuration interface",
|
|
"Select configuration interface",
|
|
config_interface_list,
|
|
0,
|
|
0);
|
|
sel_config->set_by_name(BX_DEFAULT_CONFIG_INTERFACE);
|
|
|
|
static const char *bochs_start_names[] = { "quick", "load", "edit", "run" };
|
|
|
|
// quick start option, set by command line arg
|
|
new bx_param_enum_c(menu,
|
|
"start_mode",
|
|
"Bochs start types",
|
|
"Bochs start types",
|
|
bochs_start_names,
|
|
BX_RUN_START,
|
|
BX_QUICK_START);
|
|
|
|
new bx_param_bool_c(menu,
|
|
"restore",
|
|
"Restore Bochs session",
|
|
"Restore Bochs session",
|
|
0);
|
|
new bx_param_string_c(menu,
|
|
"restore_path",
|
|
"Path to data for restore",
|
|
"Path to data for restore",
|
|
"",
|
|
BX_PATHNAME_LEN);
|
|
|
|
// benchmarking mode, set by command line arg
|
|
new bx_param_num_c(menu,
|
|
"benchmark",
|
|
"benchmark mode",
|
|
"set benchmark mode",
|
|
0, BX_MAX_BIT32U, 0);
|
|
|
|
// dump statistics, set by command line arg
|
|
new bx_param_num_c(menu,
|
|
"dumpstats",
|
|
"dumpstats mode",
|
|
"dump statistics period",
|
|
0, BX_MAX_BIT32U, 0);
|
|
// unlock disk images
|
|
new bx_param_bool_c(menu,
|
|
"unlock_images",
|
|
"Unlock disk images",
|
|
"Unlock disk images leftover previous from Bochs session",
|
|
0);
|
|
|
|
// subtree for setting up log actions by device in bochsrc
|
|
bx_list_c *logfn = new bx_list_c(menu, "logfn", "Logfunctions");
|
|
new bx_list_c(logfn, "debug", "");
|
|
new bx_list_c(logfn, "info", "");
|
|
new bx_list_c(logfn, "error", "");
|
|
new bx_list_c(logfn, "panic", "");
|
|
|
|
// optional plugin control
|
|
new bx_list_c(menu, "plugin_ctrl", "Optional Plugin Control");
|
|
bx_plugin_ctrl_init();
|
|
|
|
// subtree for special menus
|
|
bx_list_c *special_menus = new bx_list_c(root_param, "menu", "");
|
|
|
|
#if BX_SUPPORT_SMP
|
|
#define BX_CPU_PROCESSORS_LIMIT 255
|
|
#define BX_CPU_CORES_LIMIT 8
|
|
#define BX_CPU_HT_THREADS_LIMIT 4
|
|
#else
|
|
#define BX_CPU_PROCESSORS_LIMIT 1
|
|
#define BX_CPU_CORES_LIMIT 1
|
|
#define BX_CPU_HT_THREADS_LIMIT 1
|
|
#endif
|
|
|
|
// cpu subtree
|
|
bx_list_c *cpu_param = new bx_list_c(root_param, "cpu", "CPU Options");
|
|
|
|
static const char *cpu_names[] = {
|
|
#define bx_define_cpudb(model) #model,
|
|
#include "cpudb.h"
|
|
NULL
|
|
};
|
|
#undef bx_define_cpudb
|
|
|
|
new bx_param_enum_c(cpu_param,
|
|
"model", "CPU configuration",
|
|
"Choose pre-defined CPU configuration",
|
|
cpu_names, 0, 0);
|
|
|
|
// cpu options
|
|
bx_param_num_c *nprocessors = new bx_param_num_c(cpu_param,
|
|
"n_processors", "Number of processors in SMP mode",
|
|
"Sets the number of processors for multiprocessor emulation",
|
|
1, BX_CPU_PROCESSORS_LIMIT,
|
|
1);
|
|
nprocessors->set_enabled(BX_CPU_PROCESSORS_LIMIT > 1);
|
|
nprocessors->set_options(bx_param_c::CI_ONLY);
|
|
bx_param_num_c *ncores = new bx_param_num_c(cpu_param,
|
|
"n_cores", "Number of cores in each processor in SMP mode",
|
|
"Sets the number of cores per processor for multiprocessor emulation",
|
|
1, BX_CPU_CORES_LIMIT,
|
|
1);
|
|
ncores->set_enabled(BX_CPU_CORES_LIMIT > 1);
|
|
ncores->set_options(bx_param_c::CI_ONLY);
|
|
bx_param_num_c *nthreads = new bx_param_num_c(cpu_param,
|
|
"n_threads", "Number of HT threads per each core in SMP mode",
|
|
"Sets the number of HT (Intel(R) HyperThreading Technology) threads per core for multiprocessor emulation",
|
|
1, BX_CPU_HT_THREADS_LIMIT,
|
|
1);
|
|
nthreads->set_enabled(BX_CPU_HT_THREADS_LIMIT > 1);
|
|
nthreads->set_options(bx_param_c::CI_ONLY);
|
|
new bx_param_num_c(cpu_param,
|
|
"ips", "Emulated instructions per second (IPS)",
|
|
"Emulated instructions per second, used to calibrate bochs emulated time with wall clock time.",
|
|
BX_MIN_IPS, BX_MAX_BIT32U,
|
|
4000000);
|
|
#if BX_SUPPORT_SMP
|
|
new bx_param_num_c(cpu_param,
|
|
"quantum", "Quantum ticks in SMP simulation",
|
|
"Maximum amount of instructions allowed to execute before returning control to another CPU.",
|
|
BX_SMP_QUANTUM_MIN, BX_SMP_QUANTUM_MAX,
|
|
16);
|
|
#endif
|
|
new bx_param_bool_c(cpu_param,
|
|
"reset_on_triple_fault", "Enable CPU reset on triple fault",
|
|
"Enable CPU reset if triple fault occurred (highly recommended)",
|
|
1);
|
|
#if BX_CPU_LEVEL >= 5
|
|
new bx_param_bool_c(cpu_param,
|
|
"ignore_bad_msrs", "Ignore RDMSR / WRMSR to unknown MSR register",
|
|
"Ignore RDMSR/WRMSR to unknown MSR register",
|
|
1);
|
|
#endif
|
|
new bx_param_bool_c(cpu_param,
|
|
"cpuid_limit_winnt", "Limit max CPUID function to 3",
|
|
"Limit max CPUID function reported to 3 to workaround WinNT issue",
|
|
0);
|
|
#if BX_SUPPORT_MONITOR_MWAIT
|
|
new bx_param_bool_c(cpu_param,
|
|
"mwait_is_nop", "Don't put CPU to sleep state by MWAIT",
|
|
"Don't put CPU to sleep state by MWAIT",
|
|
0);
|
|
#endif
|
|
#if BX_CONFIGURE_MSRS
|
|
new bx_param_filename_c(cpu_param,
|
|
"msrs",
|
|
"Configurable MSR definition file",
|
|
"Set path to the configurable MSR definition file",
|
|
"", BX_PATHNAME_LEN);
|
|
#endif
|
|
|
|
cpu_param->set_options(menu->SHOW_PARENT);
|
|
|
|
// cpuid subtree
|
|
#if BX_CPU_LEVEL >= 4
|
|
bx_list_c *cpuid_param = new bx_list_c(root_param, "cpuid", "CPUID Options");
|
|
|
|
new bx_param_num_c(cpuid_param,
|
|
"level", "CPU Level",
|
|
"CPU level",
|
|
(BX_CPU_LEVEL < 5) ? BX_CPU_LEVEL : 5, BX_CPU_LEVEL,
|
|
BX_CPU_LEVEL);
|
|
|
|
new bx_param_num_c(cpuid_param,
|
|
"stepping", "Stepping ID",
|
|
"Processor 4-bits stepping ID",
|
|
0, 15,
|
|
3);
|
|
|
|
new bx_param_num_c(cpuid_param,
|
|
"model", "Model ID",
|
|
"Processor model ID, extended model ID",
|
|
0, 255,
|
|
3);
|
|
|
|
new bx_param_num_c(cpuid_param,
|
|
"family", "Family ID",
|
|
"Processor family ID, extended family ID",
|
|
BX_CPU_LEVEL, (BX_CPU_LEVEL >= 6) ? 4095 : BX_CPU_LEVEL,
|
|
BX_CPU_LEVEL);
|
|
|
|
new bx_param_string_c(cpuid_param,
|
|
"vendor_string",
|
|
"CPUID vendor string",
|
|
"Set the CPUID vendor string",
|
|
#if BX_CPU_VENDOR_INTEL
|
|
"GenuineIntel",
|
|
#else
|
|
"AuthenticAMD",
|
|
#endif
|
|
BX_CPUID_VENDOR_LEN+1);
|
|
new bx_param_string_c(cpuid_param,
|
|
"brand_string",
|
|
"CPUID brand string",
|
|
"Set the CPUID brand string",
|
|
#if BX_CPU_VENDOR_INTEL
|
|
" Intel(R) Pentium(R) 4 CPU ",
|
|
#else
|
|
"AMD Athlon(tm) processor",
|
|
#endif
|
|
BX_CPUID_BRAND_LEN+1);
|
|
|
|
#if BX_CPU_LEVEL >= 5
|
|
new bx_param_bool_c(cpuid_param,
|
|
"mmx", "Support for MMX instruction set",
|
|
"Support for MMX instruction set",
|
|
1);
|
|
|
|
// configure defaults to XAPIC enabled
|
|
static const char *apic_names[] = {
|
|
"legacy",
|
|
"xapic",
|
|
#if BX_CPU_LEVEL >= 6
|
|
"xapic_ext",
|
|
"x2apic",
|
|
#endif
|
|
NULL
|
|
};
|
|
|
|
new bx_param_enum_c(cpuid_param,
|
|
"apic", "APIC configuration",
|
|
"Select APIC configuration (Legacy APIC/XAPIC/XAPIC_EXT/X2APIC)",
|
|
apic_names,
|
|
BX_CPUID_SUPPORT_XAPIC,
|
|
BX_CPUID_SUPPORT_LEGACY_APIC);
|
|
#endif
|
|
|
|
#if BX_CPU_LEVEL >= 6
|
|
// configure defaults to CPU_LEVEL = 6 with SSE2 enabled
|
|
static const char *simd_names[] = {
|
|
"none",
|
|
"sse",
|
|
"sse2",
|
|
"sse3",
|
|
"ssse3",
|
|
"sse4_1",
|
|
"sse4_2",
|
|
#if BX_SUPPORT_AVX
|
|
"avx",
|
|
"avx2",
|
|
#if BX_SUPPORT_EVEX
|
|
"avx512",
|
|
#endif
|
|
#endif
|
|
NULL };
|
|
|
|
new bx_param_enum_c(cpuid_param,
|
|
"simd", "Support for SIMD instruction set",
|
|
"Support for SIMD (SSE/SSE2/SSE3/SSSE3/SSE4_1/SSE4_2/AVX/AVX2/AVX512) instruction set",
|
|
simd_names,
|
|
BX_CPUID_SUPPORT_SSE2,
|
|
BX_CPUID_SUPPORT_NOSSE);
|
|
|
|
new bx_param_bool_c(cpuid_param,
|
|
"sse4a", "Support for AMD SSE4A instructions",
|
|
"Support for AMD SSE4A instructions",
|
|
0);
|
|
new bx_param_bool_c(cpuid_param,
|
|
"misaligned_sse", "Support for AMD Misaligned SSE mode",
|
|
"Support for AMD Misaligned SSE mode",
|
|
0);
|
|
|
|
new bx_param_bool_c(cpuid_param,
|
|
"sep", "Support for SYSENTER/SYSEXIT instructions",
|
|
"Support for SYSENTER/SYSEXIT instructions",
|
|
1);
|
|
new bx_param_bool_c(cpuid_param,
|
|
"movbe", "Support for MOVBE instruction",
|
|
"Support for MOVBE instruction",
|
|
0);
|
|
new bx_param_bool_c(cpuid_param,
|
|
"adx", "Support for ADX instructions",
|
|
"Support for ADCX/ADOX instructions",
|
|
0);
|
|
new bx_param_bool_c(cpuid_param,
|
|
"aes", "Support for AES instruction set",
|
|
"Support for AES instruction set",
|
|
0);
|
|
new bx_param_bool_c(cpuid_param,
|
|
"sha", "Support for SHA instruction set",
|
|
"Support for SHA instruction set",
|
|
0);
|
|
new bx_param_bool_c(cpuid_param,
|
|
"xsave", "Support for XSAVE extensions",
|
|
"Support for XSAVE extensions",
|
|
0);
|
|
new bx_param_bool_c(cpuid_param,
|
|
"xsaveopt", "Support for XSAVEOPT instruction",
|
|
"Support for XSAVEOPT instruction",
|
|
0);
|
|
#if BX_SUPPORT_AVX
|
|
new bx_param_bool_c(cpuid_param,
|
|
"avx_f16c", "Support for AVX F16 convert instructions",
|
|
"Support for AVX F16 convert instructions",
|
|
0);
|
|
new bx_param_bool_c(cpuid_param,
|
|
"avx_fma", "Support for AVX FMA instructions",
|
|
"Support for AVX FMA instructions",
|
|
0);
|
|
new bx_param_num_c(cpuid_param,
|
|
"bmi", "Support for BMI instructions",
|
|
"Support for Bit Manipulation Instructions (BMI)",
|
|
0, 2,
|
|
0);
|
|
new bx_param_bool_c(cpuid_param,
|
|
"xop", "Support for AMD XOP instructions",
|
|
"Support for AMD XOP instructions",
|
|
0);
|
|
new bx_param_bool_c(cpuid_param,
|
|
"fma4", "Support for AMD four operand FMA instructions",
|
|
"Support for AMD FMA4 instructions",
|
|
0);
|
|
new bx_param_bool_c(cpuid_param,
|
|
"tbm", "Support for AMD TBM instructions",
|
|
"Support for AMD Trailing Bit Manipulation (TBM) instructions",
|
|
0);
|
|
#endif
|
|
#if BX_SUPPORT_X86_64
|
|
new bx_param_bool_c(cpuid_param,
|
|
"x86_64", "x86-64 and long mode",
|
|
"Support for x86-64 and long mode",
|
|
1);
|
|
new bx_param_bool_c(cpuid_param,
|
|
"1g_pages", "1G pages support in long mode",
|
|
"Support for 1G pages in long mode",
|
|
0);
|
|
new bx_param_bool_c(cpuid_param,
|
|
"pcid", "PCID support in long mode",
|
|
"Support for process context ID (PCID) in long mode",
|
|
0);
|
|
new bx_param_bool_c(cpuid_param,
|
|
"fsgsbase", "FS/GS BASE access instructions support",
|
|
"FS/GS BASE access instructions support in long mode",
|
|
0);
|
|
#endif
|
|
new bx_param_bool_c(cpuid_param,
|
|
"smep", "Supervisor Mode Execution Protection support",
|
|
"Supervisor Mode Execution Protection support",
|
|
0);
|
|
new bx_param_bool_c(cpuid_param,
|
|
"smap", "Supervisor Mode Access Prevention support",
|
|
"Supervisor Mode Access Prevention support",
|
|
0);
|
|
#if BX_SUPPORT_MONITOR_MWAIT
|
|
new bx_param_bool_c(cpuid_param,
|
|
"mwait", "MONITOR/MWAIT instructions support",
|
|
"MONITOR/MWAIT instructions support",
|
|
BX_SUPPORT_MONITOR_MWAIT);
|
|
#endif
|
|
#if BX_SUPPORT_VMX
|
|
new bx_param_num_c(cpuid_param,
|
|
"vmx", "Support for Intel VMX extensions emulation",
|
|
"Support for Intel VMX extensions emulation",
|
|
0, BX_SUPPORT_VMX,
|
|
1);
|
|
#endif
|
|
#if BX_SUPPORT_SVM
|
|
new bx_param_bool_c(cpuid_param,
|
|
"svm", "Secure Virtual Machine (SVM) emulation support",
|
|
"Secure Virtual Machine (SVM) emulation support",
|
|
0);
|
|
#endif
|
|
#endif // CPU_LEVEL >= 6
|
|
|
|
cpuid_param->set_options(menu->SHOW_PARENT | menu->USE_SCROLL_WINDOW);
|
|
|
|
// CPUID subtree depends on CPU model
|
|
SIM->get_param_enum(BXPN_CPU_MODEL)->set_dependent_list(cpuid_param->clone(), 0);
|
|
// enable CPUID subtree only for CPU model choice #0
|
|
SIM->get_param_enum(BXPN_CPU_MODEL)->set_dependent_bitmap(0, BX_MAX_BIT64U);
|
|
|
|
#endif // CPU_LEVEL >= 4
|
|
|
|
// memory subtree
|
|
bx_list_c *memory = new bx_list_c(root_param, "memory", "Memory Options");
|
|
bx_list_c *stdmem = new bx_list_c(memory, "standard", "Standard Options");
|
|
bx_list_c *optrom = new bx_list_c(memory, "optrom", "Optional ROM Images");
|
|
bx_list_c *optram = new bx_list_c(memory, "optram", "Optional RAM Images");
|
|
bx_list_c *ram = new bx_list_c(stdmem, "ram", "RAM size options");
|
|
bx_list_c *rom = new bx_list_c(stdmem, "rom", "BIOS ROM options");
|
|
bx_list_c *vgarom = new bx_list_c(stdmem, "vgarom", "VGABIOS ROM options");
|
|
|
|
// memory options (ram & rom)
|
|
bx_param_num_c *ramsize = new bx_param_num_c(ram,
|
|
"guest",
|
|
"Memory size (megabytes)",
|
|
"Amount of RAM in megabytes",
|
|
1, ((Bit64u)(1) << BX_PHY_ADDRESS_WIDTH) / (1024*1024),
|
|
BX_DEFAULT_MEM_MEGS);
|
|
ramsize->set_ask_format("Enter memory size (MB): [%d] ");
|
|
|
|
bx_param_num_c *host_ramsize = new bx_param_num_c(ram,
|
|
"host",
|
|
"Host allocated memory size (megabytes)",
|
|
"Amount of host allocated memory in megabytes",
|
|
1, 2048,
|
|
BX_DEFAULT_MEM_MEGS);
|
|
host_ramsize->set_ask_format("Enter host memory size (MB): [%d] ");
|
|
ram->set_options(ram->SERIES_ASK);
|
|
|
|
bx_param_num_c *mem_block_size = new bx_param_num_c(ram,
|
|
"block_size",
|
|
"Memory block granularity (kilobytes)",
|
|
"Granularity of host memory allocation",
|
|
4, 8192,
|
|
128);
|
|
mem_block_size->set_ask_format("Enter memory block size (KB): [%d] ");
|
|
ram->set_options(ram->SERIES_ASK);
|
|
|
|
path = new bx_param_filename_c(rom,
|
|
"file",
|
|
"ROM BIOS image",
|
|
"Pathname of ROM image to load",
|
|
"", BX_PATHNAME_LEN);
|
|
path->set_format("Name of ROM BIOS image: %s");
|
|
sprintf(name, "%s" DIRECTORY_SEPARATOR "BIOS-bochs-latest", (char *)get_builtin_variable("BXSHARE"));
|
|
path->set_initial_val(name);
|
|
bx_param_num_c *romaddr = new bx_param_num_c(rom,
|
|
"address",
|
|
"ROM BIOS address",
|
|
"The address at which the ROM image should be loaded",
|
|
0, BX_MAX_BIT32U,
|
|
0);
|
|
romaddr->set_base(16);
|
|
romaddr->set_format("0x%08x");
|
|
romaddr->set_long_format("ROM BIOS address: 0x%08x");
|
|
new bx_param_string_c(rom,
|
|
"options",
|
|
"BIOS options",
|
|
"Options for the Bochs BIOS",
|
|
"", BX_PATHNAME_LEN);
|
|
rom->set_options(rom->SERIES_ASK);
|
|
|
|
path = new bx_param_filename_c(vgarom,
|
|
"file",
|
|
"VGA BIOS image",
|
|
"Pathname of VGA ROM image to load",
|
|
"", BX_PATHNAME_LEN);
|
|
path->set_format("Name of VGA BIOS image: %s");
|
|
sprintf(name, "%s" DIRECTORY_SEPARATOR "VGABIOS-lgpl-latest", get_builtin_variable("BXSHARE"));
|
|
path->set_initial_val(name);
|
|
vgarom->set_options(vgarom->SERIES_ASK);
|
|
|
|
bx_list_c *optnum;
|
|
bx_param_num_c *optaddr;
|
|
|
|
for (i=0; i<BX_N_OPTROM_IMAGES; i++) {
|
|
sprintf(name, "%d", i+1);
|
|
sprintf(descr, "Pathname of optional ROM image #%d to load", i+1);
|
|
sprintf(label, "Optional ROM image #%d", i+1);
|
|
optnum = new bx_list_c(optrom, name, label);
|
|
path = new bx_param_filename_c(optnum,
|
|
"file",
|
|
"Path",
|
|
descr,
|
|
"", BX_PATHNAME_LEN);
|
|
sprintf(label, "Name of optional ROM image #%d", i+1);
|
|
strcat(label, " : %s");
|
|
path->set_format(label);
|
|
sprintf(descr, "The address at which the optional ROM image #%d should be loaded", i+1);
|
|
optaddr = new bx_param_num_c(optnum,
|
|
"address",
|
|
"Address",
|
|
descr,
|
|
0, BX_MAX_BIT32U,
|
|
0);
|
|
optaddr->set_base(16);
|
|
optaddr->set_format("0x%05x");
|
|
sprintf(label, "Optional ROM #%d address:", i+1);
|
|
strcat(label, " 0x%05x");
|
|
optaddr->set_long_format(label);
|
|
deplist = new bx_list_c(NULL);
|
|
deplist->add(optaddr);
|
|
path->set_dependent_list(deplist);
|
|
optnum->set_options(optnum->SERIES_ASK | optnum->USE_BOX_TITLE);
|
|
}
|
|
optrom->set_options(optrom->SHOW_PARENT);
|
|
|
|
for (i=0; i<BX_N_OPTRAM_IMAGES; i++) {
|
|
sprintf(name, "%d", i+1);
|
|
sprintf(descr, "Pathname of optional RAM image #%d to load", i+1);
|
|
sprintf(label, "Optional RAM image #%d", i+1);
|
|
optnum = new bx_list_c(optram, name, label);
|
|
path = new bx_param_filename_c(optnum,
|
|
"file",
|
|
"Path",
|
|
descr,
|
|
"", BX_PATHNAME_LEN);
|
|
sprintf(label, "Name of optional RAM image #%d", i+1);
|
|
strcat(label, " : %s");
|
|
path->set_format(label);
|
|
sprintf(descr, "The address at which the optional RAM image #%d should be loaded", i+1);
|
|
optaddr = new bx_param_num_c(optnum,
|
|
"address",
|
|
"Address",
|
|
descr,
|
|
0, BX_MAX_BIT32U,
|
|
0);
|
|
optaddr->set_base(16);
|
|
optaddr->set_format("0x%05x");
|
|
sprintf(label, "Optional RAM #%d address:", i+1);
|
|
strcat(label, " 0x%05x");
|
|
optaddr->set_long_format(label);
|
|
deplist = new bx_list_c(NULL);
|
|
deplist->add(optaddr);
|
|
path->set_dependent_list(deplist);
|
|
optnum->set_options(optnum->SERIES_ASK | optnum->USE_BOX_TITLE);
|
|
}
|
|
optram->set_options(optram->SHOW_PARENT);
|
|
memory->set_options(memory->SHOW_PARENT | memory->USE_TAB_WINDOW);
|
|
|
|
// clock & cmos subtree
|
|
bx_list_c *clock_cmos = new bx_list_c(root_param, "clock_cmos", "Clock & CMOS Options");
|
|
|
|
// clock & cmos options
|
|
static const char *clock_sync_names[] = { "none", "realtime", "slowdown", "both", NULL };
|
|
Bit64s mintime,maxtime;
|
|
struct tm mtim,xtim;
|
|
|
|
mtim.tm_year =0000 -1900;
|
|
mtim.tm_mon = 1 - 1;
|
|
mtim.tm_mday = 1 ;
|
|
mtim.tm_hour = 00 ;
|
|
mtim.tm_min = 00 ;
|
|
mtim.tm_sec = 00 ;
|
|
mtim.tm_isdst = -1 ;
|
|
|
|
xtim.tm_year =9999 -1900;
|
|
xtim.tm_mon = 12 - 1;
|
|
xtim.tm_mday = 31 ;
|
|
xtim.tm_hour = 23 ;
|
|
xtim.tm_min = 59 ;
|
|
xtim.tm_sec = 59 ;
|
|
xtim.tm_isdst = -1 ;
|
|
|
|
mintime=mktime(&mtim); //Find which epoch corresponds to the upper limit expressed in local time
|
|
maxtime=mktime(&xtim); //Find which epoch corresponds to the lower limit expressed in local time
|
|
|
|
bx_param_enum_c *clock_sync = new bx_param_enum_c(clock_cmos,
|
|
"clock_sync", "Synchronisation method",
|
|
"Host to guest time synchronization method",
|
|
clock_sync_names,
|
|
BX_CLOCK_SYNC_NONE,
|
|
BX_CLOCK_SYNC_NONE);
|
|
bx_param_num_c *time0 = new bx_param_num_c(clock_cmos,
|
|
"time0",
|
|
"Initial CMOS time for Bochs\n(1:localtime, 2:utc, other:time in seconds)",
|
|
"Initial time for Bochs CMOS clock, used if you really want two runs to be identical",
|
|
mintime, maxtime,
|
|
BX_CLOCK_TIME0_LOCAL);
|
|
bx_param_bool_c *rtc_sync = new bx_param_bool_c(clock_cmos,
|
|
"rtc_sync", "Sync RTC speed with realtime",
|
|
"If enabled, the RTC runs at realtime speed",
|
|
0);
|
|
deplist = new bx_list_c(NULL);
|
|
deplist->add(rtc_sync);
|
|
clock_sync->set_dependent_list(deplist, 0);
|
|
clock_sync->set_dependent_bitmap(BX_CLOCK_SYNC_REALTIME, 1);
|
|
clock_sync->set_dependent_bitmap(BX_CLOCK_SYNC_BOTH, 1);
|
|
|
|
bx_list_c *cmosimage = new bx_list_c(clock_cmos, "cmosimage", "CMOS Image Options");
|
|
bx_param_bool_c *use_cmosimage = new bx_param_bool_c(cmosimage,
|
|
"enabled", "Use a CMOS image",
|
|
"Controls the usage of a CMOS image",
|
|
0);
|
|
path = new bx_param_filename_c(cmosimage,
|
|
"path", "Pathname of CMOS image",
|
|
"Pathname of CMOS image",
|
|
"", BX_PATHNAME_LEN);
|
|
bx_param_bool_c *rtc_init = new bx_param_bool_c(cmosimage,
|
|
"rtc_init", "Initialize RTC from image",
|
|
"Controls whether to initialize the RTC with values stored in the image",
|
|
0);
|
|
deplist = new bx_list_c(NULL);
|
|
deplist->add(path);
|
|
deplist->add(rtc_init);
|
|
use_cmosimage->set_dependent_list(deplist);
|
|
|
|
time0->set_ask_format("Enter Initial CMOS time (1:localtime, 2:utc, other:time in seconds): [" FMT_LL "d] ");
|
|
clock_sync->set_ask_format("Enter Synchronisation method: [%s] ");
|
|
clock_cmos->set_options(clock_cmos->SHOW_PARENT);
|
|
cmosimage->set_options(cmosimage->SHOW_PARENT);
|
|
|
|
// pci subtree
|
|
bx_list_c *pci = new bx_list_c(root_param, "pci", "PCI Options");
|
|
|
|
// pci options
|
|
static const char *pci_chipset_names[] = { "i430fx", "i440fx", "i440bx", NULL };
|
|
deplist = new bx_list_c(NULL);
|
|
|
|
enabled = new bx_param_bool_c(pci,
|
|
"enabled",
|
|
"Enable PCI Support",
|
|
"Controls whether to emulate a PCI chipset",
|
|
BX_SUPPORT_PCI);
|
|
bx_param_enum_c *pci_chipset = new bx_param_enum_c(pci,
|
|
"chipset", "PCI chipset",
|
|
"Select PCI chipset to emulate",
|
|
pci_chipset_names,
|
|
BX_PCI_CHIPSET_I440FX,
|
|
BX_PCI_CHIPSET_I430FX);
|
|
deplist->add(pci_chipset);
|
|
// pci slots
|
|
bx_init_pcidev_list();
|
|
bx_list_c *slot = new bx_list_c(pci, "slot", "PCI Slots");
|
|
deplist->add(slot);
|
|
for (i=0; i<BX_N_PCI_SLOTS; i++) {
|
|
sprintf(name, "%d", i+1);
|
|
sprintf (descr, "Name of the device connected to PCI slot #%d", i+1);
|
|
sprintf (label, "PCI slot #%d device", i+1);
|
|
bx_param_enum_c *devname = new bx_param_enum_c(slot,
|
|
name,
|
|
label,
|
|
descr,
|
|
pcislot_dev_list,
|
|
0, 0);
|
|
deplist->add(devname);
|
|
}
|
|
bx_param_string_c *advopts = new bx_param_string_c(pci, "advopts", "Advanced PCI Options",
|
|
"Set advanced PCI options",
|
|
"", BX_PATHNAME_LEN);
|
|
deplist->add(advopts);
|
|
enabled->set_dependent_list(deplist);
|
|
enabled->set_enabled(BX_SUPPORT_PCI);
|
|
pci->set_options(pci->SHOW_PARENT);
|
|
slot->set_options(slot->SHOW_PARENT);
|
|
|
|
// display subtree
|
|
bx_list_c *display = new bx_list_c(root_param, "display", "Bochs Display & Interface Options");
|
|
|
|
bx_init_displaylib_list();
|
|
bx_param_enum_c *sel_displaylib = new bx_param_enum_c(display,
|
|
"display_library", "VGA Display Library",
|
|
"Select VGA Display Library",
|
|
display_library_list,
|
|
0,
|
|
0);
|
|
sel_displaylib->set_by_name(BX_DEFAULT_DISPLAY_LIBRARY);
|
|
sel_displaylib->set_ask_format("Choose which library to use for the Bochs display: [%s] ");
|
|
|
|
new bx_param_string_c(display,
|
|
"displaylib_options", "Display Library options",
|
|
"Options passed to Display Library",
|
|
"",
|
|
BX_PATHNAME_LEN);
|
|
|
|
new bx_param_bool_c(display,
|
|
"private_colormap", "Use a private colormap",
|
|
"Request that the GUI create and use it's own non-shared colormap. This colormap will be used when in the bochs window. If not enabled, a shared colormap scheme may be used. Not implemented on all GUI's.",
|
|
0);
|
|
|
|
#if BX_WITH_AMIGAOS
|
|
bx_param_bool_c *fullscreen = new bx_param_bool_c(display,
|
|
"fullscreen", "Use full screen mode",
|
|
"When enabled, bochs occupies the whole screen instead of just a window.",
|
|
0);
|
|
bx_param_string_c *screenmode = new bx_param_string_c(display,
|
|
"screenmode",
|
|
"Screen mode name",
|
|
"Screen mode name",
|
|
"", BX_PATHNAME_LEN);
|
|
screenmode->set_handler(bx_param_string_handler);
|
|
#endif
|
|
|
|
new bx_param_bool_c(display,
|
|
"vga_realtime",
|
|
"VGA timer realtime",
|
|
"If enabled, the VGA timer is based on realtime",
|
|
1);
|
|
|
|
bx_param_num_c *vga_update_freq = new bx_param_num_c(display,
|
|
"vga_update_frequency",
|
|
"VGA Update Frequency",
|
|
"Number of VGA updates per emulated second",
|
|
1, 60,
|
|
5);
|
|
vga_update_freq->set_ask_format ("Type a new value for VGA update frequency: [%d] ");
|
|
|
|
bx_init_vgaext_list();
|
|
bx_param_enum_c *vga_extension = new bx_param_enum_c(display,
|
|
"vga_extension",
|
|
"VGA Extension",
|
|
"Name of the VGA extension",
|
|
vga_extension_names,
|
|
0, 0);
|
|
vga_extension->set_by_name("vbe");
|
|
vga_extension->set_handler(bx_param_handler);
|
|
display->set_options(display->SHOW_PARENT);
|
|
|
|
static const char *ddc_mode_list[] = {
|
|
"disabled",
|
|
"builtin",
|
|
"file",
|
|
NULL
|
|
};
|
|
bx_param_enum_c *ddc_mode = new bx_param_enum_c(display,
|
|
"ddc_mode", "DDC emulation mode",
|
|
"Select DDC emulation mode",
|
|
ddc_mode_list,
|
|
BX_DDC_MODE_BUILTIN,
|
|
BX_DDC_MODE_DISABLED);
|
|
path = new bx_param_filename_c(display,
|
|
"ddc_file",
|
|
"DDC definition file",
|
|
"Set path to a DDC definition file",
|
|
"", BX_PATHNAME_LEN);
|
|
deplist = new bx_list_c(NULL);
|
|
deplist->add(path);
|
|
ddc_mode->set_dependent_list(deplist, 0);
|
|
ddc_mode->set_dependent_bitmap(BX_DDC_MODE_FILE, 1);
|
|
|
|
// keyboard & mouse subtree
|
|
bx_list_c *kbd_mouse = new bx_list_c(root_param, "keyboard_mouse", "Keyboard & Mouse Options");
|
|
bx_list_c *keyboard = new bx_list_c(kbd_mouse, "keyboard", "Keyboard Options");
|
|
bx_list_c *mouse = new bx_list_c(kbd_mouse, "mouse", "Mouse Options");
|
|
|
|
static const char *keyboard_type_names[] = { "xt", "at", "mf", NULL };
|
|
|
|
// keyboard & mouse options
|
|
type = new bx_param_enum_c(keyboard,
|
|
"type", "Keyboard type",
|
|
"Keyboard type reported by the 'identify keyboard' command",
|
|
keyboard_type_names,
|
|
BX_KBD_MF_TYPE,
|
|
BX_KBD_XT_TYPE);
|
|
type->set_ask_format ("Enter keyboard type: [%s] ");
|
|
|
|
new bx_param_num_c(keyboard,
|
|
"serial_delay", "Keyboard serial delay",
|
|
"Approximate time in microseconds that it takes one character to be transferred from the keyboard to controller over the serial path.",
|
|
5, BX_MAX_BIT32U,
|
|
250);
|
|
new bx_param_num_c(keyboard,
|
|
"paste_delay", "Keyboard paste delay",
|
|
"Approximate time in microseconds between attempts to paste characters to the keyboard controller.",
|
|
1000, BX_MAX_BIT32U,
|
|
100000);
|
|
bx_param_bool_c *use_kbd_mapping = new bx_param_bool_c(keyboard,
|
|
"use_mapping", "Use keyboard mapping",
|
|
"Controls whether to use the keyboard mapping feature",
|
|
0);
|
|
use_kbd_mapping->set_options(bx_param_c::CI_ONLY);
|
|
bx_param_filename_c *keymap = new bx_param_filename_c(keyboard,
|
|
"keymap", "Keymap filename",
|
|
"Pathname of the keymap file used",
|
|
"", BX_PATHNAME_LEN);
|
|
keymap->set_extension("map");
|
|
deplist = new bx_list_c(NULL);
|
|
deplist->add(keymap);
|
|
use_kbd_mapping->set_dependent_list(deplist);
|
|
|
|
bx_param_string_c *user_shortcut = new bx_param_string_c(keyboard,
|
|
"user_shortcut",
|
|
"Userbutton shortcut",
|
|
"Defines the keyboard shortcut to be sent when you press the 'user' button in the headerbar.",
|
|
"none", 20);
|
|
user_shortcut->set_handler(bx_param_string_handler);
|
|
|
|
static const char *mouse_type_list[] = {
|
|
"none",
|
|
"ps2",
|
|
"imps2",
|
|
#if BX_SUPPORT_BUSMOUSE
|
|
"inport",
|
|
"bus",
|
|
#endif
|
|
"serial",
|
|
"serial_wheel",
|
|
"serial_msys",
|
|
NULL
|
|
};
|
|
type = new bx_param_enum_c(mouse,
|
|
"type", "Mouse type",
|
|
"The mouse type can be one of these: 'none', 'ps2', 'imps2', 'serial', 'serial_wheel'"
|
|
#if BX_SUPPORT_BUSMOUSE
|
|
", 'bus'"
|
|
#endif
|
|
,
|
|
mouse_type_list,
|
|
BX_MOUSE_TYPE_PS2,
|
|
BX_MOUSE_TYPE_NONE);
|
|
type->set_ask_format("Choose the type of mouse [%s] ");
|
|
|
|
new bx_param_bool_c(mouse,
|
|
"enabled", "Enable mouse capture",
|
|
"Controls whether the mouse sends events to the guest. The hardware emulation is always enabled.",
|
|
0);
|
|
|
|
static const char *mouse_toggle_list[] = {
|
|
"ctrl+mbutton",
|
|
"ctrl+f10",
|
|
"ctrl+alt",
|
|
"f12",
|
|
NULL
|
|
};
|
|
toggle = new bx_param_enum_c(mouse,
|
|
"toggle", "Mouse toggle method",
|
|
"The mouse toggle method can be one of these: 'ctrl+mbutton', 'ctrl+f10', 'ctrl+alt'",
|
|
mouse_toggle_list,
|
|
BX_MOUSE_TOGGLE_CTRL_MB,
|
|
BX_MOUSE_TOGGLE_CTRL_MB);
|
|
toggle->set_ask_format("Choose the mouse toggle method [%s] ");
|
|
|
|
kbd_mouse->set_options(kbd_mouse->SHOW_PARENT);
|
|
keyboard->set_options(keyboard->SHOW_PARENT);
|
|
mouse->set_options(mouse->SHOW_PARENT);
|
|
|
|
// boot parameter subtree
|
|
bx_list_c *boot_params = new bx_list_c(root_param, "boot_params", "Boot Options");
|
|
// boot sequence
|
|
for (i=0; i<3; i++) {
|
|
sprintf(name, "boot_drive%d", i+1);
|
|
sprintf(label, "Boot drive #%d", i+1);
|
|
sprintf(descr, "Name of drive #%d in boot sequence (A, C or CD)", i+1);
|
|
bx_param_enum_c *bootdrive = new bx_param_enum_c(boot_params,
|
|
name,
|
|
label,
|
|
descr,
|
|
&bochs_bootdisk_names[(i==0)?BX_BOOT_FLOPPYA:BX_BOOT_NONE],
|
|
(i==0)?BX_BOOT_FLOPPYA:BX_BOOT_NONE,
|
|
(i==0)?BX_BOOT_FLOPPYA:BX_BOOT_NONE);
|
|
bootdrive->set_ask_format("Boot from floppy drive, hard drive or cdrom ? [%s] ");
|
|
}
|
|
|
|
new bx_param_bool_c(boot_params,
|
|
"floppy_sig_check",
|
|
"Skip Floppy Boot Signature Check",
|
|
"Skips check for the 0xaa55 signature on floppy boot device.",
|
|
0);
|
|
|
|
boot_params->set_options(menu->SHOW_PARENT);
|
|
|
|
// floppy subtree
|
|
bx_list_c *floppy = new bx_list_c(root_param, "floppy", "Floppy Options");
|
|
new bx_list_c(floppy, "0", "First Floppy Drive");
|
|
new bx_list_c(floppy, "1", "Second Floppy Drive");
|
|
|
|
bx_param_enum_c *devtype;
|
|
// floppy options
|
|
for (i = 0; i < 2; i++) {
|
|
|
|
bx_list_c *floppyX = (bx_list_c*)floppy->get(i);
|
|
|
|
devtype = new bx_param_enum_c(floppyX,
|
|
"devtype",
|
|
"Type of floppy drive",
|
|
"Type of floppy drive",
|
|
floppy_devtype_names,
|
|
(i==0)?BX_FDD_350HD:BX_FDD_NONE,
|
|
BX_FDD_NONE);
|
|
devtype->set_ask_format("What type of floppy drive? [%s] ");
|
|
devtype->set_handler(bx_param_handler);
|
|
|
|
if (i == 0) {
|
|
strcpy(label, "First floppy image/device");
|
|
strcpy(descr, "Pathname of first floppy image file or device. If you're booting from floppy, this should be a bootable floppy.");
|
|
} else {
|
|
strcpy(label, "Second floppy image/device");
|
|
strcpy(descr, "Pathname of second floppy image file or device.");
|
|
}
|
|
path = new bx_param_filename_c(floppyX, "path", label, descr, "", BX_PATHNAME_LEN);
|
|
path->set_ask_format("Enter new filename, or 'none' for no disk: [%s] ");
|
|
path->set_extension("img");
|
|
path->set_initial_val("none");
|
|
|
|
type = new bx_param_enum_c(floppyX,
|
|
"type",
|
|
"Type of floppy media",
|
|
"Type of floppy media",
|
|
floppy_type_names,
|
|
BX_FLOPPY_NONE,
|
|
BX_FLOPPY_NONE);
|
|
type->set_ask_format("What type of floppy media? (auto=detect) [%s] ");
|
|
type->set_handler(bx_param_handler);
|
|
type->set_runtime_param(1);
|
|
|
|
readonly = new bx_param_bool_c(floppyX,
|
|
"readonly",
|
|
"Write Protection",
|
|
"Floppy media write protection",
|
|
0);
|
|
readonly->set_ask_format("Is media write protected? [%s] ");
|
|
|
|
status = new bx_param_enum_c(floppyX,
|
|
"status",
|
|
"Status",
|
|
"Floppy media status (inserted / ejected)",
|
|
media_status_names,
|
|
BX_EJECTED,
|
|
BX_EJECTED);
|
|
status->set_ask_format("Is the device inserted or ejected? [%s] ");
|
|
|
|
deplist = new bx_list_c(NULL);
|
|
deplist->add(path);
|
|
devtype->set_dependent_list(deplist, 1);
|
|
devtype->set_dependent_bitmap(BX_FDD_NONE, 0);
|
|
|
|
deplist = new bx_list_c(NULL);
|
|
deplist->add(type);
|
|
deplist->add(readonly);
|
|
deplist->add(status);
|
|
path->set_dependent_list(deplist);
|
|
|
|
floppyX->set_options(floppyX->SERIES_ASK | floppyX->USE_BOX_TITLE);
|
|
}
|
|
|
|
floppy->set_options(floppy->SHOW_PARENT);
|
|
|
|
// ATA/ATAPI subtree
|
|
bx_hdimage_ctl.init(); // initialize available disk image modes
|
|
bx_list_c *ata = new bx_list_c(root_param, "ata", "ATA/ATAPI Options");
|
|
ata->set_options(ata->USE_TAB_WINDOW);
|
|
|
|
// disk options
|
|
const char *s_atachannel[] = {
|
|
"ATA channel 0",
|
|
"ATA channel 1",
|
|
"ATA channel 2",
|
|
"ATA channel 3",
|
|
};
|
|
const char *s_atadevname[2] = {
|
|
"master",
|
|
"slave",
|
|
};
|
|
const char *s_atadevice[4][2] = {
|
|
{ "First HD/CD on channel 0",
|
|
"Second HD/CD on channel 0" },
|
|
{ "First HD/CD on channel 1",
|
|
"Second HD/CD on channel 1" },
|
|
{ "First HD/CD on channel 2",
|
|
"Second HD/CD on channel 2" },
|
|
{ "First HD/CD on channel 3",
|
|
"Second HD/CD on channel 3" }
|
|
};
|
|
Bit16u ata_default_ioaddr1[4] = {
|
|
0x1f0, 0x170, 0x1e8, 0x168
|
|
};
|
|
Bit16u ata_default_ioaddr2[4] = {
|
|
0x3f0, 0x370, 0x3e0, 0x360
|
|
};
|
|
Bit8u ata_default_irq[4] = {
|
|
14, 15, 11, 9
|
|
};
|
|
|
|
#define BXP_PARAMS_PER_ATA_DEVICE 12
|
|
|
|
bx_list_c *ata_menu[BX_MAX_ATA_CHANNEL];
|
|
bx_list_c *ata_res[BX_MAX_ATA_CHANNEL];
|
|
|
|
for (Bit8u channel=0; channel<BX_MAX_ATA_CHANNEL; channel++) {
|
|
|
|
sprintf(name, "%d", channel);
|
|
ata_menu[channel] = new bx_list_c(ata, name, s_atachannel[channel]);
|
|
ata_menu[channel]->set_options(bx_list_c::USE_TAB_WINDOW);
|
|
ata_res[channel] = new bx_list_c(ata_menu[channel], "resources", s_atachannel[channel]);
|
|
ata_res[channel]->set_options(bx_list_c::SERIES_ASK);
|
|
|
|
enabled = new bx_param_bool_c(ata_res[channel],
|
|
"enabled",
|
|
"Enable ATA channel",
|
|
"Controls whether ata channel is installed or not",
|
|
0);
|
|
enabled->set_ask_format("Channel is enabled: [%s] ");
|
|
|
|
ioaddr = new bx_param_num_c(ata_res[channel],
|
|
"ioaddr1",
|
|
"I/O Address 1",
|
|
"IO address of ata command block",
|
|
0, 0xffff,
|
|
ata_default_ioaddr1[channel]);
|
|
ioaddr->set_base(16);
|
|
ioaddr->set_ask_format("Enter new ioaddr1: [0x%x] ");
|
|
|
|
ioaddr2 = new bx_param_num_c(ata_res[channel],
|
|
"ioaddr2",
|
|
"I/O Address 2",
|
|
"IO address of ata control block",
|
|
0, 0xffff,
|
|
ata_default_ioaddr2[channel]);
|
|
ioaddr2->set_base(16);
|
|
ioaddr2->set_ask_format("Enter new ioaddr2: [0x%x] ");
|
|
|
|
irq = new bx_param_num_c(ata_res[channel],
|
|
"irq",
|
|
"IRQ",
|
|
"IRQ used by this ata channel",
|
|
0, 15,
|
|
ata_default_irq[channel]);
|
|
irq->set_ask_format("Enter new IRQ: [%d] ");
|
|
irq->set_options(irq->USE_SPIN_CONTROL);
|
|
|
|
// all items in the ata[channel] menu depend on the enabled flag.
|
|
// The menu list is complete, but a few dependent_list items will
|
|
// be added later. Use clone() to make a copy of the dependent_list
|
|
// so that it can be changed without affecting the menu.
|
|
enabled->set_dependent_list(ata_res[channel]->clone());
|
|
|
|
for (Bit8u slave=0; slave<2; slave++) {
|
|
menu = new bx_list_c(ata_menu[channel],
|
|
s_atadevname[slave],
|
|
s_atadevice[channel][slave]);
|
|
menu->set_options(menu->SERIES_ASK);
|
|
|
|
static const char *atadevice_type_names[] = { "none", "disk", "cdrom", NULL };
|
|
|
|
type = new bx_param_enum_c(menu,
|
|
"type",
|
|
"Type of ATA device",
|
|
"Type of ATA device (disk or cdrom)",
|
|
atadevice_type_names,
|
|
BX_ATA_DEVICE_NONE,
|
|
BX_ATA_DEVICE_NONE);
|
|
type->set_ask_format("Enter type of ATA device, disk or cdrom: [%s] ");
|
|
|
|
path = new bx_param_filename_c(menu,
|
|
"path",
|
|
"Path or physical device name",
|
|
"Pathname of the image or physical device (cdrom only)",
|
|
"", BX_PATHNAME_LEN);
|
|
path->set_ask_format("Enter new filename: [%s] ");
|
|
path->set_extension("img");
|
|
|
|
mode = new bx_param_enum_c(menu,
|
|
"mode",
|
|
"Type of disk image",
|
|
"Mode of the ATA harddisk",
|
|
bx_hdimage_ctl.get_mode_names(),
|
|
0, 0);
|
|
mode->set_ask_format("Enter mode of ATA device, (flat, concat, etc.): [%s] ");
|
|
|
|
status = new bx_param_enum_c(menu,
|
|
"status",
|
|
"Status",
|
|
"CD-ROM media status (inserted / ejected)",
|
|
media_status_names,
|
|
BX_EJECTED,
|
|
BX_EJECTED);
|
|
status->set_ask_format("Is the device inserted or ejected? [%s] ");
|
|
|
|
bx_param_filename_c *journal = new bx_param_filename_c(menu,
|
|
"journal",
|
|
"Path of journal file",
|
|
"Pathname of the journal file",
|
|
"", BX_PATHNAME_LEN);
|
|
journal->set_ask_format("Enter path of journal file: [%s]");
|
|
deplist = new bx_list_c(NULL);
|
|
deplist->add(journal);
|
|
mode->set_dependent_list(deplist, 0);
|
|
mode->set_dependent_bitmap(bx_hdimage_ctl.get_mode_id("undoable"), 1);
|
|
mode->set_dependent_bitmap(bx_hdimage_ctl.get_mode_id("volatile"), 1);
|
|
mode->set_dependent_bitmap(bx_hdimage_ctl.get_mode_id("vvfat"), 1);
|
|
|
|
bx_param_num_c *cylinders = new bx_param_num_c(menu,
|
|
"cylinders",
|
|
"Cylinders",
|
|
"Number of cylinders",
|
|
0, 262143,
|
|
0);
|
|
cylinders->set_ask_format("Enter number of cylinders: [%d] ");
|
|
bx_param_num_c *heads = new bx_param_num_c(menu,
|
|
"heads",
|
|
"Heads",
|
|
"Number of heads",
|
|
0, 16,
|
|
0);
|
|
heads->set_ask_format("Enter number of heads: [%d] ");
|
|
bx_param_num_c *spt = new bx_param_num_c(menu,
|
|
"spt",
|
|
"Sectors per track",
|
|
"Number of sectors per track",
|
|
0, 63,
|
|
0);
|
|
spt->set_ask_format("Enter number of sectors per track: [%d] ");
|
|
|
|
static const char *sector_size_names[] = { "512", "1024", "4096", NULL };
|
|
|
|
bx_param_enum_c *sect_size = new bx_param_enum_c(menu,
|
|
"sect_size",
|
|
"Sector size",
|
|
"Size of a disk sector in bytes",
|
|
sector_size_names,
|
|
BX_SECT_SIZE_512,
|
|
BX_SECT_SIZE_512);
|
|
sect_size->set_ask_format("Enter sector size: [%s]");
|
|
|
|
bx_param_string_c *model = new bx_param_string_c(menu,
|
|
"model",
|
|
"Model name",
|
|
"String returned by the 'identify device' command",
|
|
"Generic 1234", 41);
|
|
model->set_ask_format("Enter new model name: [%s]");
|
|
|
|
static const char *atadevice_biosdetect_names[] = { "auto", "cmos", "none", NULL };
|
|
|
|
bx_param_enum_c *biosdetect = new bx_param_enum_c(menu,
|
|
"biosdetect",
|
|
"BIOS Detection",
|
|
"Type of bios detection",
|
|
atadevice_biosdetect_names,
|
|
BX_ATA_BIOSDETECT_AUTO,
|
|
BX_ATA_BIOSDETECT_AUTO);
|
|
biosdetect->set_ask_format("Enter bios detection type: [%s]");
|
|
|
|
static const char *atadevice_translation_names[] = { "none", "lba", "large", "rechs", "auto", NULL };
|
|
|
|
bx_param_enum_c *translation = new bx_param_enum_c(menu,
|
|
"translation",
|
|
"Translation type",
|
|
"How the ata-disk translation is done by the bios",
|
|
atadevice_translation_names,
|
|
BX_ATA_TRANSLATION_AUTO,
|
|
BX_ATA_TRANSLATION_NONE);
|
|
translation->set_ask_format("Enter translation type: [%s]");
|
|
|
|
// the master/slave menu depends on the ATA channel's enabled flag
|
|
enabled->get_dependent_list()->add(menu);
|
|
// the type selector depends on the ATA channel's enabled flag
|
|
enabled->get_dependent_list()->add(type);
|
|
|
|
// all items depend on the drive type
|
|
type->set_dependent_list(menu->clone(), 0);
|
|
type->set_dependent_bitmap(BX_ATA_DEVICE_DISK, 0xfe6);
|
|
type->set_dependent_bitmap(BX_ATA_DEVICE_CDROM, 0x60a);
|
|
|
|
type->set_handler(bx_param_handler);
|
|
}
|
|
|
|
// Enable two ATA interfaces by default, disable the others.
|
|
// Now that the dependence relationships are established, call set() on
|
|
// the ata device present params to set all enables correctly.
|
|
enabled->set_initial_val(channel<2);
|
|
enabled->set(channel<2);
|
|
}
|
|
|
|
// disk menu
|
|
bx_param_c *disk_menu_init_list[] = {
|
|
SIM->get_param(BXPN_FLOPPYA),
|
|
SIM->get_param(BXPN_FLOPPYB),
|
|
SIM->get_param(BXPN_ATA0_RES),
|
|
SIM->get_param(BXPN_ATA0_MASTER),
|
|
SIM->get_param(BXPN_ATA0_SLAVE),
|
|
#if BX_MAX_ATA_CHANNEL>1
|
|
SIM->get_param(BXPN_ATA1_RES),
|
|
SIM->get_param(BXPN_ATA1_MASTER),
|
|
SIM->get_param(BXPN_ATA1_SLAVE),
|
|
#endif
|
|
#if BX_MAX_ATA_CHANNEL>2
|
|
SIM->get_param(BXPN_ATA2_RES),
|
|
SIM->get_param(BXPN_ATA2_MASTER),
|
|
SIM->get_param(BXPN_ATA2_SLAVE),
|
|
#endif
|
|
#if BX_MAX_ATA_CHANNEL>3
|
|
SIM->get_param(BXPN_ATA3_RES),
|
|
SIM->get_param(BXPN_ATA3_MASTER),
|
|
SIM->get_param(BXPN_ATA3_SLAVE),
|
|
#endif
|
|
SIM->get_param("boot_params"),
|
|
NULL
|
|
};
|
|
menu = new bx_list_c(special_menus, "disk", "Bochs Disk Options", disk_menu_init_list);
|
|
menu->set_options(menu->SHOW_PARENT);
|
|
|
|
#ifdef WIN32
|
|
// disk menu for win32paramdlg
|
|
bx_param_c *disk_menu2_init_list[] = {
|
|
SIM->get_param("floppy"),
|
|
SIM->get_param("ata.0"),
|
|
#if BX_MAX_ATA_CHANNEL>1
|
|
SIM->get_param("ata.1"),
|
|
#endif
|
|
#if BX_MAX_ATA_CHANNEL>2
|
|
SIM->get_param("ata.2"),
|
|
#endif
|
|
#if BX_MAX_ATA_CHANNEL>3
|
|
SIM->get_param("ata.3"),
|
|
#endif
|
|
SIM->get_param("boot_params"),
|
|
NULL
|
|
};
|
|
menu = new bx_list_c(special_menus, "disk_win32", "Bochs Disk Options", disk_menu2_init_list);
|
|
menu->set_options(menu->USE_TAB_WINDOW);
|
|
#endif
|
|
|
|
// ports subtree
|
|
bx_list_c *ports = new bx_list_c(root_param, "ports", "Serial / Parallel / USB Options");
|
|
ports->set_options(ports->USE_TAB_WINDOW | ports->SHOW_PARENT);
|
|
#if BX_SUPPORT_PCIUSB
|
|
bx_usbdev_ctl.init();
|
|
#endif
|
|
// parallel / serial / USB options initialized in the device plugin code
|
|
|
|
#if BX_NETWORKING
|
|
// network subtree
|
|
bx_list_c *network = new bx_list_c(root_param, "network", "Network Configuration");
|
|
network->set_options(network->USE_TAB_WINDOW | network->SHOW_PARENT);
|
|
bx_netmod_ctl.init();
|
|
// network device options initialized in the device plugin code
|
|
#endif
|
|
|
|
// sound subtree
|
|
bx_list_c *sound = new bx_list_c(root_param, "sound", "Sound Configuration");
|
|
sound->set_options(sound->USE_TAB_WINDOW | sound->SHOW_PARENT);
|
|
#if BX_SUPPORT_SOUNDLOW
|
|
bx_list_c *soundlow = new bx_list_c(sound, "lowlevel", "Lowlevel Sound Configuration");
|
|
soundlow->set_options(soundlow->SHOW_PARENT | soundlow->SERIES_ASK);
|
|
soundlow->set_enabled(BX_SUPPORT_SOUNDLOW);
|
|
|
|
bx_soundmod_ctl.init();
|
|
bx_param_enum_c *driver = new bx_param_enum_c(soundlow,
|
|
"waveoutdrv",
|
|
"Waveout driver",
|
|
"This is the waveout driver to use for emulated sound devices",
|
|
bx_soundmod_ctl.get_driver_names(),
|
|
0, 0);
|
|
driver->set_by_name(BX_SOUND_LOWLEVEL_NAME);
|
|
new bx_param_filename_c(soundlow,
|
|
"waveout",
|
|
"Wave output device",
|
|
"This is the device where the wave output is sent to",
|
|
"", BX_PATHNAME_LEN);
|
|
driver = new bx_param_enum_c(soundlow,
|
|
"waveindrv",
|
|
"Wavein driver",
|
|
"This is the wavein driver to use for emulated sound devices",
|
|
bx_soundmod_ctl.get_driver_names(),
|
|
0, 0);
|
|
driver->set_by_name(BX_SOUND_LOWLEVEL_NAME);
|
|
new bx_param_filename_c(soundlow,
|
|
"wavein",
|
|
"Wave input device",
|
|
"This is the device to be used as the wave input source",
|
|
"", BX_PATHNAME_LEN);
|
|
driver = new bx_param_enum_c(soundlow,
|
|
"midioutdrv",
|
|
"Midiout driver",
|
|
"This is the midiout driver to use for emulated sound devices",
|
|
bx_soundmod_ctl.get_driver_names(),
|
|
0, 0);
|
|
driver->set_by_name(BX_SOUND_LOWLEVEL_NAME);
|
|
new bx_param_filename_c(soundlow,
|
|
"midiout",
|
|
"MIDI output device",
|
|
"This is the device where the MIDI output is sent to",
|
|
"", BX_PATHNAME_LEN);
|
|
#endif
|
|
// sound device options initialized in the device plugin code
|
|
|
|
// misc options subtree
|
|
bx_list_c *misc = new bx_list_c(root_param, "misc", "Configure Everything Else");
|
|
misc->set_options(misc->SHOW_PARENT);
|
|
|
|
// port e9 hack
|
|
bx_list_c *port_e9_hack = new bx_list_c(misc, "port_e9_hack", "port 0xE9 hack");
|
|
new bx_param_bool_c(port_e9_hack,
|
|
"enabled",
|
|
"Enable port 0xE9 hack",
|
|
"Debug messages written to i/o port 0xE9 will be displayed on console",
|
|
0);
|
|
|
|
// port e9 hack all rings
|
|
new bx_param_bool_c(port_e9_hack,
|
|
"all_rings",
|
|
"Enable port 0xE9 hack for all rings",
|
|
"Debug messages written to i/o port 0xE9 from ring3 will be displayed on console",
|
|
0);
|
|
|
|
#if BX_SUPPORT_IODEBUG
|
|
// iodebug all rings
|
|
new bx_param_bool_c(misc,
|
|
"iodebug_all_rings",
|
|
"Enable iodebug ports for all rings",
|
|
"I/O Interface to Bochs Debugger plugin for all rings",
|
|
0);
|
|
#endif
|
|
|
|
// GDB stub
|
|
menu = new bx_list_c(misc, "gdbstub", "GDB Stub Options");
|
|
menu->set_options(menu->SHOW_PARENT | menu->USE_BOX_TITLE);
|
|
menu->set_enabled(BX_GDBSTUB);
|
|
enabled = new bx_param_bool_c(menu,
|
|
"enabled",
|
|
"Enable GDB stub",
|
|
"",
|
|
0);
|
|
enabled->set_enabled(BX_GDBSTUB);
|
|
new bx_param_num_c(menu,
|
|
"port",
|
|
"Port",
|
|
"TCP/IP port for GDB stub",
|
|
0, 65535,
|
|
1234);
|
|
new bx_param_num_c(menu,
|
|
"text_base",
|
|
"Text base",
|
|
"",
|
|
0, BX_MAX_BIT32U,
|
|
0);
|
|
new bx_param_num_c(menu,
|
|
"data_base",
|
|
"Data base",
|
|
"",
|
|
0, BX_MAX_BIT32U,
|
|
0);
|
|
new bx_param_num_c(menu,
|
|
"bss_base",
|
|
"BSS base",
|
|
"",
|
|
0, BX_MAX_BIT32U,
|
|
0);
|
|
enabled->set_dependent_list(menu->clone());
|
|
|
|
#if BX_PLUGINS
|
|
// user-defined options subtree
|
|
bx_list_c *user = new bx_list_c(root_param, "user", "User-defined options");
|
|
user->set_options(user->SHOW_PARENT);
|
|
#endif
|
|
|
|
// log options subtree
|
|
menu = new bx_list_c(root_param, "log", "Logfile Options");
|
|
menu->set_options(menu->SHOW_PARENT);
|
|
|
|
// log options
|
|
path = new bx_param_filename_c(menu,
|
|
"filename",
|
|
"Log filename",
|
|
"Pathname of bochs log file",
|
|
"-", BX_PATHNAME_LEN);
|
|
path->set_ask_format("Enter log filename: [%s] ");
|
|
path->set_extension("txt");
|
|
|
|
bx_param_string_c *prefix = new bx_param_string_c(menu,
|
|
"prefix",
|
|
"Log output prefix",
|
|
"Prefix prepended to log output",
|
|
"%t%e%d", BX_LOGPREFIX_LEN);
|
|
prefix->set_ask_format("Enter log prefix: [%s] ");
|
|
|
|
path = new bx_param_filename_c(menu,
|
|
"debugger_filename",
|
|
"Debugger Log filename",
|
|
"Pathname of debugger log file",
|
|
"-", BX_PATHNAME_LEN);
|
|
path->set_ask_format("Enter debugger log filename: [%s] ");
|
|
path->set_extension("log");
|
|
path->set_enabled(BX_DEBUGGER);
|
|
|
|
// runtime options
|
|
menu = new bx_list_c(special_menus, "runtime", "Runtime options");
|
|
bx_list_c *cdrom = new bx_list_c(menu, "cdrom", "CD-ROM options");
|
|
cdrom->set_runtime_param(1);
|
|
cdrom->set_options(cdrom->SHOW_PARENT);
|
|
// misc runtime options
|
|
misc = new bx_list_c(menu, "misc", "Misc options");
|
|
misc->set_runtime_param(1);
|
|
misc->add(SIM->get_param(BXPN_VGA_UPDATE_FREQUENCY));
|
|
misc->add(SIM->get_param(BXPN_MOUSE_ENABLED));
|
|
misc->add(SIM->get_param(BXPN_KBD_PASTE_DELAY));
|
|
misc->add(SIM->get_param(BXPN_USER_SHORTCUT));
|
|
misc->add(SIM->get_param(BXPN_PORT_E9_HACK));
|
|
misc->add(SIM->get_param(BXPN_PORT_E9_HACK_ALL_RINGS));
|
|
#if BX_SUPPORT_IODEBUG
|
|
misc->add(SIM->get_param(BXPN_IODEBUG_ALL_RINGS));
|
|
#endif
|
|
misc->set_options(misc->SHOW_PARENT | misc->SHOW_GROUP_NAME);
|
|
}
|
|
|
|
void bx_reset_options()
|
|
{
|
|
// optional plugin control
|
|
bx_plugin_ctrl_reset(1);
|
|
|
|
// cpu
|
|
SIM->get_param("cpu")->reset();
|
|
|
|
#if BX_CPU_LEVEL >= 4
|
|
// cpuid
|
|
SIM->get_param("cpuid")->reset();
|
|
#endif
|
|
|
|
// memory (ram & rom)
|
|
SIM->get_param("memory")->reset();
|
|
|
|
// clock & cmos
|
|
SIM->get_param("clock_cmos")->reset();
|
|
|
|
// pci
|
|
SIM->get_param("pci")->reset();
|
|
|
|
// display & interface
|
|
SIM->get_param("display")->reset();
|
|
|
|
// keyboard & mouse
|
|
SIM->get_param("keyboard_mouse")->reset();
|
|
|
|
// boot
|
|
SIM->get_param("boot_params")->reset();
|
|
|
|
// floppy drives
|
|
SIM->get_param("floppy")->reset();
|
|
|
|
// ata/atapi drives
|
|
SIM->get_param("ata")->reset();
|
|
|
|
// serial/parallel/usb
|
|
SIM->get_param("ports")->reset();
|
|
|
|
// network devices
|
|
SIM->get_param("network")->reset();
|
|
|
|
// sound devices
|
|
SIM->get_param("sound")->reset();
|
|
|
|
// misc
|
|
SIM->get_param("misc")->reset();
|
|
|
|
// logfile
|
|
SIM->get_param("log")->reset();
|
|
|
|
#if BX_PLUGINS
|
|
// user-defined options
|
|
SIM->get_param("user")->reset();
|
|
#endif
|
|
}
|
|
|
|
void bx_cleanup_options()
|
|
{
|
|
free(config_interface_list);
|
|
free(display_library_list);
|
|
free(vga_extension_names);
|
|
free(vga_extension_plugins);
|
|
free(pcislot_dev_list);
|
|
}
|
|
|
|
int bx_read_configuration(const char *rcfile)
|
|
{
|
|
// parse rcfile first, then parse arguments in order.
|
|
BX_INFO (("reading configuration from %s", rcfile));
|
|
if (parse_bochsrc(rcfile) < 0) {
|
|
BX_PANIC (("reading from %s failed", rcfile));
|
|
return -1;
|
|
}
|
|
// update log actions
|
|
for (int level=0; level<N_LOGLEV; level++) {
|
|
int action = SIM->get_default_log_action(level);
|
|
io->set_log_action(level, action);
|
|
}
|
|
bx_set_log_actions_by_device(0);
|
|
return 0;
|
|
}
|
|
|
|
int bx_parse_cmdline(int arg, int argc, char *argv[])
|
|
{
|
|
int level, def_action[N_LOGLEV];
|
|
|
|
for (level=0; level<N_LOGLEV; level++) {
|
|
def_action[level] = SIM->get_default_log_action(level);
|
|
}
|
|
while (arg < argc) {
|
|
BX_INFO (("parsing arg %d, %s", arg, argv[arg]));
|
|
parse_line_unformatted("cmdline args", argv[arg]);
|
|
arg++;
|
|
}
|
|
// update log actions if default has been changed
|
|
for (level=0; level<N_LOGLEV; level++) {
|
|
int action = SIM->get_default_log_action(level);
|
|
if (action != def_action[level]) {
|
|
io->set_log_action(level, action);
|
|
}
|
|
}
|
|
bx_set_log_actions_by_device(0);
|
|
return 0;
|
|
}
|
|
|
|
char *bx_find_bochsrc()
|
|
{
|
|
FILE *fd = NULL;
|
|
char rcfile[512];
|
|
Bit32u retry = 0, found = 0;
|
|
// try several possibilities for the bochsrc before giving up
|
|
while (!found) {
|
|
rcfile[0] = 0;
|
|
switch (retry++) {
|
|
case 0: strcpy (rcfile, ".bochsrc"); break;
|
|
case 1: strcpy (rcfile, "bochsrc"); break;
|
|
case 2: strcpy (rcfile, "bochsrc.txt"); break;
|
|
#ifdef WIN32
|
|
case 3: strcpy (rcfile, "bochsrc.bxrc"); break;
|
|
#elif !BX_WITH_MACOS
|
|
// only try this on unix
|
|
case 3:
|
|
{
|
|
char *ptr = getenv("HOME");
|
|
if (ptr) snprintf (rcfile, sizeof(rcfile), "%s/.bochsrc", ptr);
|
|
}
|
|
break;
|
|
case 4: strcpy (rcfile, "/etc/bochsrc"); break;
|
|
#endif
|
|
default:
|
|
return NULL;
|
|
}
|
|
if (rcfile[0]) {
|
|
BX_DEBUG (("looking for configuration in %s", rcfile));
|
|
fd = fopen(rcfile, "r");
|
|
if (fd) found = 1;
|
|
}
|
|
}
|
|
assert (fd != NULL && rcfile[0] != 0);
|
|
fclose (fd);
|
|
return strdup(rcfile);
|
|
}
|
|
|
|
static int parse_bochsrc(const char *rcfile)
|
|
{
|
|
FILE *fd = NULL;
|
|
char *ret;
|
|
char line[512];
|
|
char context[BX_PATHNAME_LEN];
|
|
Bit32u linenum = 1;
|
|
|
|
// try several possibilities for the bochsrc before giving up
|
|
|
|
bochsrc_include_level++;
|
|
|
|
fd = fopen (rcfile, "r");
|
|
if (fd == NULL) return -1;
|
|
|
|
int retval = 0;
|
|
do {
|
|
ret = fgets(line, sizeof(line)-1, fd);
|
|
line[sizeof(line) - 1] = '\0';
|
|
size_t len = strlen(line);
|
|
if ((len>0) && (line[len-1] < ' '))
|
|
line[len-1] = '\0';
|
|
if ((ret != NULL) && strlen(line)) {
|
|
sprintf(context, "%s:%u", rcfile, linenum);
|
|
if (parse_line_unformatted(context, line) < 0) {
|
|
retval = -1;
|
|
break; // quit parsing after first error
|
|
}
|
|
}
|
|
linenum++;
|
|
} while (!feof(fd));
|
|
fclose(fd);
|
|
bochsrc_include_level--;
|
|
return retval;
|
|
}
|
|
|
|
const char *get_builtin_variable(const char *varname)
|
|
{
|
|
#ifdef WIN32
|
|
int code;
|
|
DWORD size;
|
|
DWORD type = 0;
|
|
HKEY hkey;
|
|
char keyname[80];
|
|
static char data[MAX_PATH];
|
|
#endif
|
|
|
|
if (strlen(varname)<1) return NULL;
|
|
else {
|
|
if (!strcmp(varname, "BXSHARE")) {
|
|
#ifdef WIN32
|
|
wsprintf(keyname, "Software\\Bochs");
|
|
code = RegOpenKeyEx(HKEY_LOCAL_MACHINE, keyname, 0, KEY_READ, &hkey);
|
|
if (code == ERROR_SUCCESS) {
|
|
data[0] = 0;
|
|
size = MAX_PATH;
|
|
if (RegQueryValueEx(hkey, "", NULL, (LPDWORD)&type, (LPBYTE)data,
|
|
(LPDWORD)&size) == ERROR_SUCCESS) {
|
|
RegCloseKey(hkey);
|
|
return data;
|
|
} else {
|
|
RegCloseKey(hkey);
|
|
return NULL;
|
|
}
|
|
} else {
|
|
return NULL;
|
|
}
|
|
#else
|
|
return BX_SHARE_PATH;
|
|
#endif
|
|
}
|
|
return NULL;
|
|
}
|
|
}
|
|
|
|
static int parse_line_unformatted(const char *context, char *line)
|
|
{
|
|
#define MAX_PARAMS_LEN 40
|
|
char *ptr;
|
|
unsigned i, string_i = 0;
|
|
char string[512];
|
|
char *params[MAX_PARAMS_LEN];
|
|
int num_params;
|
|
bool inquotes = 0;
|
|
bool comment = 0;
|
|
|
|
memset(params, 0, sizeof(params));
|
|
if (line == NULL) return 0;
|
|
|
|
// if passed nothing but whitespace, just return
|
|
for (i=0; i<strlen(line); i++) {
|
|
if (!isspace(line[i])) break;
|
|
}
|
|
if (i>=strlen(line))
|
|
return 0;
|
|
|
|
num_params = 0;
|
|
|
|
if (!strncmp(line, "#include", 8))
|
|
ptr = strtok(line, " ");
|
|
else
|
|
ptr = strtok(line, ":");
|
|
while ((ptr) && (!comment)) {
|
|
if (!inquotes) {
|
|
string_i = 0;
|
|
} else {
|
|
string[string_i++] = ',';
|
|
}
|
|
for (i=0; i<strlen(ptr); i++) {
|
|
if (ptr[i] == '"')
|
|
inquotes = !inquotes;
|
|
else if ((ptr[i] == '#') && (strncmp(line+i, "#include", 8)) && !inquotes) {
|
|
comment = 1;
|
|
break;
|
|
} else {
|
|
#if BX_HAVE_GETENV
|
|
// substitute environment variables.
|
|
if (ptr[i] == '$') {
|
|
char varname[512];
|
|
char *pv = varname;
|
|
const char *value;
|
|
*pv = 0;
|
|
i++;
|
|
while (isalpha(ptr[i]) || ptr[i]=='_') {
|
|
*pv = ptr[i]; pv++; i++;
|
|
}
|
|
*pv = 0;
|
|
if (strlen(varname)<1 || !(value = getenv(varname))) {
|
|
if ((value = get_builtin_variable(varname))) {
|
|
if ((string_i + strlen(value)) < 512) {
|
|
// append value to the string
|
|
for (pv=(char *)value; *pv; pv++)
|
|
string[string_i++] = *pv;
|
|
} else {
|
|
BX_PANIC(("parse_line_unformatted(): out of memory"));
|
|
}
|
|
} else {
|
|
BX_PANIC(("could not look up environment variable '%s'", varname));
|
|
}
|
|
} else {
|
|
if ((string_i + strlen(value)) < 512) {
|
|
// append value to the string
|
|
for (pv=(char *)value; *pv; pv++)
|
|
string[string_i++] = *pv;
|
|
} else {
|
|
BX_PANIC(("parse_line_unformatted(): out of memory"));
|
|
}
|
|
}
|
|
}
|
|
#endif
|
|
if (!isspace(ptr[i]) || inquotes) {
|
|
if (string_i < 511) {
|
|
string[string_i++] = ptr[i];
|
|
} else {
|
|
BX_PANIC(("parse_line_unformatted(): out of memory"));
|
|
}
|
|
}
|
|
}
|
|
}
|
|
string[string_i] = '\0';
|
|
if (string_i == 0) break;
|
|
if (!inquotes) {
|
|
if (params[num_params] != NULL) {
|
|
free(params[num_params]);
|
|
params[num_params] = NULL;
|
|
}
|
|
if (num_params < MAX_PARAMS_LEN) {
|
|
params[num_params++] = strdup(string);
|
|
} else {
|
|
BX_PANIC (("too many parameters, max is %d\n", MAX_PARAMS_LEN));
|
|
}
|
|
}
|
|
ptr = strtok(NULL, ",");
|
|
}
|
|
Bit32s retval = parse_line_formatted(context, num_params, ¶ms[0]);
|
|
for (i=0; i < MAX_PARAMS_LEN; i++)
|
|
{
|
|
if (params[i] != NULL)
|
|
{
|
|
free(params[i]);
|
|
params[i] = NULL;
|
|
}
|
|
}
|
|
return retval;
|
|
}
|
|
|
|
// These macros are called for all parse errors, so that we can easily
|
|
// change the behavior of all occurrences.
|
|
#define PARSE_ERR(x) \
|
|
do { BX_PANIC(x); return -1; } while (0)
|
|
#define PARSE_WARN(x) \
|
|
BX_ERROR(x)
|
|
|
|
/*
|
|
* this supports the "floppyx: image=" option.
|
|
* the functions returns the type of the floppy
|
|
* image (1.44, 360, etc.), based on the image file size.
|
|
*/
|
|
int get_floppy_type_from_image(const char *filename)
|
|
{
|
|
struct stat stat_buf;
|
|
|
|
if (!strncmp(filename, "vvfat:", 6)) {
|
|
return BX_FLOPPY_1_44;
|
|
} else if (stat(filename, &stat_buf)) {
|
|
return BX_FLOPPY_NONE;
|
|
} else {
|
|
switch (stat_buf.st_size) {
|
|
case 163840:
|
|
return BX_FLOPPY_160K;
|
|
case 184320:
|
|
return BX_FLOPPY_180K;
|
|
case 327680:
|
|
return BX_FLOPPY_320K;
|
|
case 368640:
|
|
return BX_FLOPPY_360K;
|
|
case 737280:
|
|
return BX_FLOPPY_720K;
|
|
case 1228800:
|
|
return BX_FLOPPY_1_2;
|
|
case 1474560:
|
|
case 1720320:
|
|
case 1763328:
|
|
case 1884160:
|
|
return BX_FLOPPY_1_44;
|
|
case 2949120:
|
|
return BX_FLOPPY_2_88;
|
|
default:
|
|
return BX_FLOPPY_UNKNOWN;
|
|
}
|
|
}
|
|
}
|
|
|
|
static Bit32s parse_log_options(const char *context, int num_params, char *params[])
|
|
{
|
|
int level, action, i;
|
|
bool def_action = 0;
|
|
char *param, *module, *actstr;
|
|
char pname[20];
|
|
bx_list_c *base;
|
|
bx_param_num_c *mparam;
|
|
|
|
if (!strcmp(params[0], "panic")) {
|
|
level = LOGLEV_PANIC;
|
|
} else if (!strcmp(params[0], "error")) {
|
|
level = LOGLEV_ERROR;
|
|
} else if (!strcmp(params[0], "info")) {
|
|
level = LOGLEV_INFO;
|
|
} else { /* debug */
|
|
level = LOGLEV_DEBUG;
|
|
}
|
|
for (i = 1; i < num_params; i++) {
|
|
param = strdup(params[i]);
|
|
module = strtok(param, "=");
|
|
actstr = strtok(NULL, "");
|
|
if (actstr != NULL) {
|
|
def_action = !strcmp(module, "action");
|
|
action = SIM->is_action_name(actstr);
|
|
if (action < ACT_IGNORE) {
|
|
PARSE_ERR(("%s: %s directive malformed.", context, params[0]));
|
|
free(param);
|
|
return -1;
|
|
}
|
|
// exclude some action / level combinations (see siminterface.h)
|
|
if (BX_LOG_OPTS_EXCLUDE(level, action)) {
|
|
PARSE_ERR(("%s: event type '%s' does not support log action '%s'.", context, params[0], actstr));
|
|
free(param);
|
|
return -1;
|
|
}
|
|
if (def_action) {
|
|
SIM->set_default_log_action(level, action);
|
|
} else {
|
|
sprintf(pname, "general.logfn.%s", params[0]);
|
|
base = (bx_list_c*) SIM->get_param(pname);
|
|
mparam = (bx_param_num_c*) base->get_by_name(module);
|
|
if (mparam != NULL) {
|
|
mparam->set(action);
|
|
} else {
|
|
mparam = new bx_param_num_c(base, module, "", "", -1, BX_MAX_BIT32U, action);
|
|
if (mparam == NULL) {
|
|
PARSE_ERR(("%s: %s: failed to add log module.", context, params[0]));
|
|
}
|
|
}
|
|
}
|
|
} else {
|
|
PARSE_ERR(("%s: %s directive malformed.", context, params[0]));
|
|
free(param);
|
|
return -1;
|
|
}
|
|
free(param);
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
static int parse_debug_symbols(const char *context, const char **params, int num_params)
|
|
{
|
|
#if BX_DEBUGGER
|
|
Bit32u offset = 0;
|
|
const char *filename = 0;
|
|
|
|
while (num_params > 0)
|
|
{
|
|
if (!strncmp(*params, "file=", 5)) {
|
|
filename = *params + 5;
|
|
}
|
|
else if (!strncmp(*params, "offset=", 7)) {
|
|
char* end;
|
|
offset = strtoul(*params + 7, &end, 0);
|
|
if (*end)
|
|
PARSE_ERR(("%s: debug_symbols: invalid parameter %s", context, *params));
|
|
}
|
|
else {
|
|
PARSE_ERR(("%s: debug_symbols: invalid parameter %s", context, *params));
|
|
}
|
|
params++; num_params--;
|
|
}
|
|
|
|
if (!filename)
|
|
PARSE_ERR(("%s: debug_symbols: missing file name", context));
|
|
|
|
if (bx_dbg_symbol_command(filename, 1, offset) < 0)
|
|
PARSE_ERR(("%s: debug_symbols: failed to load symbols from '%s'", context, filename));
|
|
#endif
|
|
return 0;
|
|
}
|
|
|
|
static int parse_param_bool(const char *input, int len, const char *param)
|
|
{
|
|
if (SIM->get_param_bool(param)->parse_param(&input[len]) == 1) {
|
|
return 0;
|
|
}
|
|
|
|
return -1;
|
|
}
|
|
|
|
int bx_parse_param_from_list(const char *context, const char *input, bx_list_c *list)
|
|
{
|
|
char *propval, *property, *value;
|
|
bx_param_c *param;
|
|
int ret = 0;
|
|
|
|
if (list == NULL) {
|
|
PARSE_WARN(("%s: parameter list == NULL!", context));
|
|
return -1;
|
|
}
|
|
propval = strdup(input);
|
|
property = strtok(propval, "=");
|
|
value = strtok(NULL, "");
|
|
if (!strcmp(property, input)) {
|
|
PARSE_WARN(("%s: incorrect parameter format", context));
|
|
free(propval);
|
|
return -1;
|
|
}
|
|
param = list->get_by_name(property);
|
|
if (param != NULL) {
|
|
if ((param->get_options() & param->CI_ONLY) > 0) {
|
|
PARSE_WARN(("%s: ignoring hidden parameter '%s'", context, property));
|
|
free(propval);
|
|
return 0;
|
|
}
|
|
int res = param->parse_param(value);
|
|
if (res != -1) {
|
|
if (res == 0) {
|
|
PARSE_WARN(("%s: wrong value for parameter '%s'", context, property));
|
|
ret = -1;
|
|
}
|
|
}
|
|
else {
|
|
PARSE_WARN(("%s: parameter '%s': unknown type", context, property));
|
|
ret = -1;
|
|
}
|
|
} else {
|
|
PARSE_WARN(("%s: unknown parameter '%s'", context, property));
|
|
ret = -1;
|
|
}
|
|
free(propval);
|
|
return ret;
|
|
}
|
|
|
|
int bx_parse_usb_port_params(const char *context, const char *param,
|
|
int maxports, bx_list_c *base)
|
|
{
|
|
bool devopt = 0;
|
|
int idx, plen;
|
|
char tmpname[20], newopts[BX_PATHNAME_LEN];
|
|
char *devstr, *arg, *pEnd;
|
|
const char *opt = NULL, *origopts;
|
|
static bool compat_mode = false;
|
|
|
|
if (!strncmp(param, "port", 4)) {
|
|
devopt = 1;
|
|
plen = 4;
|
|
} else {
|
|
devopt = 0;
|
|
plen = 7;
|
|
}
|
|
// this gets the long int number value after the string: 'port'.
|
|
// Ex: 'port15=' will produce 15 for idx. We allow up to 30 for now.
|
|
idx = (int) strtol(¶m[plen], &pEnd, 10);
|
|
if ((idx < 1) || (idx > 30) || (pEnd[0] != '=')) {
|
|
PARSE_ERR(("%s: usb_%s: portX / optionsX parameter malformed.", context, base->get_name()));
|
|
return -1;
|
|
}
|
|
if (idx > maxports) {
|
|
PARSE_ERR(("%s: usb_%s: port number out of range.", context, base->get_name()));
|
|
return -1;
|
|
}
|
|
sprintf(tmpname, "port%d.%s", idx, devopt ? "device" : "options");
|
|
if (devopt) {
|
|
compat_mode = false;
|
|
|
|
// if we already have found this port's declaration in the bochsrc.txt file, give error
|
|
// for example:
|
|
// usb_ohci: port1=mouse, options1="speed:low, model:m228"
|
|
// usb_ohci: port1=keyboard, options1="speed:low"
|
|
// this catches the second line, giving an error...
|
|
if (SIM->get_param_enum(tmpname, base)->get() > 0) {
|
|
BX_PANIC(("%s: Already declared port%d type. Please choose another available port number.", context, idx));
|
|
}
|
|
|
|
if (!SIM->get_param_enum(tmpname, base)->set_by_name(¶m[plen + 2])) {
|
|
// backward compatibility code
|
|
devstr = strdup(¶m[plen + 2]);
|
|
arg = strtok(devstr, ":");
|
|
arg = strtok(NULL, "\n");
|
|
SIM->get_param_enum(tmpname, base)->set_by_name(devstr);
|
|
if (arg != NULL) {
|
|
if (!strcmp(devstr, "disk") || !strcmp(devstr, "cdrom") ||
|
|
!strcmp(devstr, "floppy")) {
|
|
opt = "path";
|
|
} else if (!strcmp(devstr, "hub")) {
|
|
opt = "ports";
|
|
} else if (!strcmp(devstr, "printer")) {
|
|
opt = "file";
|
|
}
|
|
if (opt != NULL) {
|
|
sprintf(tmpname, "port%d.options", idx);
|
|
origopts = SIM->get_param_string(tmpname, base)->getptr();
|
|
if (strlen(origopts) > 0) {
|
|
sprintf(newopts, "%s:%s, %s", opt, arg, origopts);
|
|
} else {
|
|
sprintf(newopts, "%s:%s", opt, arg);
|
|
}
|
|
SIM->get_param_string(tmpname, base)->set(newopts);
|
|
compat_mode = true;
|
|
}
|
|
}
|
|
free(devstr);
|
|
}
|
|
} else {
|
|
// if we already have found this options#= declaration in the bochsrc.txt file, give warning.
|
|
// (it is legal to do this, as in:
|
|
// usb_ohci: port2=disk
|
|
// usb_ohci: options2="speed:full"
|
|
// usb_ohci: options2="path:hdd.img"
|
|
// however, this catches something like:
|
|
// usb_ohci: port1=tablet, options1="speed:low"
|
|
// usb_ohci: port2=disk, options1="speed:full, path:hdd.img"
|
|
// where the user copy/pasted something and forgot to adjust the options# in the second line)
|
|
if (!SIM->get_param_string(tmpname, base)->isempty()) {
|
|
BX_INFO(("%s: Already declared options%d parameter. Was this intended?", context, idx));
|
|
}
|
|
|
|
if (compat_mode) {
|
|
origopts = SIM->get_param_string(tmpname, base)->getptr();
|
|
sprintf(newopts, "%s, %s", origopts, ¶m[plen + 2]);
|
|
compat_mode = false;
|
|
} else {
|
|
strcpy(newopts, ¶m[plen + 2]);
|
|
}
|
|
SIM->get_param_string(tmpname, base)->set(newopts);
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
int bx_parse_nic_params(const char *context, const char *param, bx_list_c *base)
|
|
{
|
|
int valid = 0;
|
|
int n;
|
|
bx_param_bytestring_c *bsp;
|
|
|
|
if (!strncmp(param, "enabled=", 8)) {
|
|
SIM->get_param_bool("enabled", base)->parse_param(¶m[8]);
|
|
n = SIM->get_param_bool("enabled", base)->get();
|
|
valid &= 0x3f;
|
|
if (n == 0) valid |= 0x80;
|
|
else valid |= 0x40;
|
|
} else if (!strncmp(param, "mac=", 4)) {
|
|
bsp = (bx_param_bytestring_c*)SIM->get_param_string("mac", base);
|
|
if (bsp->parse_param(¶m[4]) == 0) {
|
|
PARSE_ERR(("%s: '%s' mac address malformed.", context, base->get_name()));
|
|
} else {
|
|
valid |= 0x04;
|
|
}
|
|
} else if (!strncmp(param, "ethmod=", 7)) {
|
|
if (!SIM->get_param_enum("ethmod", base)->set_by_name(¶m[7]))
|
|
PARSE_ERR(("%s: ethernet module '%s' not available", context, ¶m[7]));
|
|
} else if (bx_parse_param_from_list(context, param, base) < 0) {
|
|
PARSE_WARN(("%s: expected parameter '%s' for '%s' ignored.", context, param, base->get_name()));
|
|
return -1;
|
|
}
|
|
return valid;
|
|
}
|
|
|
|
int bx_split_option_list(const char *msg, const char *rawopt, char **argv, int max_argv)
|
|
{
|
|
char *ptr, *ptr2, *tmpstr;
|
|
int argc = 0, i;
|
|
|
|
char *options = new char[strlen(rawopt)+1];
|
|
strcpy(options, rawopt);
|
|
ptr = strtok(options, ",");
|
|
while (ptr && strcmp(ptr, "none")) {
|
|
if (argc < max_argv) {
|
|
tmpstr = new char[strlen(ptr)+1];
|
|
strcpy(tmpstr, ptr);
|
|
ptr2 = tmpstr;
|
|
while (isspace(*ptr2)) ptr2++;
|
|
i = (int)strlen(ptr2) - 1;
|
|
while ((i >= 0) && isspace(ptr2[i])) {
|
|
ptr2[i] = 0;
|
|
i--;
|
|
}
|
|
if (strlen(ptr2) > 0) {
|
|
argv[argc++] = strdup(ptr2);
|
|
}
|
|
delete [] tmpstr;
|
|
} else {
|
|
BX_ERROR(("%s: too many parameters, max is %d", msg, max_argv));
|
|
}
|
|
ptr = strtok(NULL, ",");
|
|
}
|
|
delete [] options;
|
|
return argc;
|
|
}
|
|
|
|
bool is_deprecated_option(const char *oldparam, const char **newparam)
|
|
{
|
|
if ((!strcmp(oldparam, "keyboard_serial_delay")) ||
|
|
(!strcmp(oldparam, "keyboard_paste_delay")) ||
|
|
(!strcmp(oldparam, "keyboard_type")) ||
|
|
(!strcmp(oldparam, "keyboard_mapping")) ||
|
|
(!strcmp(oldparam, "keyboardmapping"))) {
|
|
// replaced v2.6 / removed v2.6.7
|
|
*newparam = "keyboard";
|
|
return 1;
|
|
} else if (!strcmp(oldparam, "user_shortcut")) {
|
|
// replaced v2.6.1 / removed v2.6.9
|
|
*newparam = "keyboard";
|
|
return 1;
|
|
#if BX_SUPPORT_PCIPNIC
|
|
} else if (!strcmp(oldparam, "pnic")) {
|
|
// replaced v2.6 / removed v2.6.5
|
|
*newparam = "pcipnic";
|
|
return 1;
|
|
#endif
|
|
#if BX_PLUGINS
|
|
} else if (!strcmp(oldparam, "user_plugin")) {
|
|
// replaced / removed after v2.6.11
|
|
*newparam = "plugin_ctrl";
|
|
return 1;
|
|
#endif
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
static int parse_line_formatted(const char *context, int num_params, char *params[])
|
|
{
|
|
int i, slot, t, dt;
|
|
bx_list_c *base;
|
|
const char *newparam;
|
|
char *value;
|
|
|
|
if (num_params < 1) return 0;
|
|
if (num_params < 2) {
|
|
PARSE_ERR(("%s: a bochsrc option needs at least one parameter", context));
|
|
}
|
|
|
|
if (!strcmp(params[0], "#include")) {
|
|
if (num_params != 2) {
|
|
PARSE_ERR(("%s: ignoring malformed #include directive.", context));
|
|
}
|
|
if (!strcmp(params[1], context)) {
|
|
PARSE_ERR(("%s: cannot include this file again.", context));
|
|
}
|
|
if (bochsrc_include_level > 2) {
|
|
PARSE_ERR(("%s: maximum include level exceeded (limit = 2).", context));
|
|
}
|
|
bx_read_configuration(params[1]);
|
|
} else if (!strcmp(params[0], "plugin_ctrl")) {
|
|
char *param, *pname, *val;
|
|
for (i=1; i<num_params; i++) {
|
|
param = strdup(params[i]);
|
|
pname = strtok(param, "=");
|
|
val = strtok(NULL, "");
|
|
if (val != NULL) {
|
|
if (!strcmp(val, "0") || !stricmp(val, "false")) {
|
|
SIM->opt_plugin_ctrl(pname, 0);
|
|
} else if (!strcmp(val, "1") || !stricmp(val, "true")) {
|
|
SIM->opt_plugin_ctrl(pname, 1);
|
|
} else {
|
|
PARSE_ERR(("%s: plugin_ctrl directive malformed", context));
|
|
}
|
|
} else {
|
|
PARSE_ERR(("%s: plugin_ctrl directive malformed", context));
|
|
}
|
|
free(param);
|
|
}
|
|
} else if (!strcmp(params[0], "config_interface")) {
|
|
if (num_params != 2) {
|
|
PARSE_ERR(("%s: config_interface directive: wrong # args.", context));
|
|
}
|
|
if (!SIM->get_param_enum(BXPN_SEL_CONFIG_INTERFACE)->set_by_name(params[1]))
|
|
PARSE_ERR(("%s: config_interface '%s' not available", context, params[1]));
|
|
} else if (!strcmp(params[0], "display_library")) {
|
|
if ((num_params < 2) || (num_params > 3)) {
|
|
PARSE_ERR(("%s: display_library directive: wrong # args.", context));
|
|
}
|
|
if (SIM->get_param_enum(BXPN_SEL_DISPLAY_LIBRARY)->set_by_name(params[1])) {
|
|
i = 2;
|
|
} else {
|
|
i = 1;
|
|
}
|
|
if ((num_params == 3) || (i == 1)) {
|
|
if (!strncmp(params[i], "options=", 8)) {
|
|
SIM->get_param_string(BXPN_DISPLAYLIB_OPTIONS)->set(¶ms[i][8]);
|
|
} else if (i == 1) {
|
|
PARSE_ERR(("%s: display library '%s' not available", context, params[1]));
|
|
} else {
|
|
PARSE_ERR(("%s: display_library directive malformed", context));
|
|
}
|
|
}
|
|
} else if ((!strcmp(params[0], "floppya")) ||
|
|
(!strcmp(params[0], "floppyb"))) {
|
|
if (!strcmp(params[0], "floppya")) {
|
|
base = (bx_list_c*) SIM->get_param(BXPN_FLOPPYA);
|
|
} else {
|
|
base = (bx_list_c*) SIM->get_param(BXPN_FLOPPYB);
|
|
}
|
|
for (i=1; i<num_params; i++) {
|
|
if (!strncmp(params[i], "type=", 5)) {
|
|
value = params[i] + 5;
|
|
dt = -1;
|
|
if (!strcmp(value, "2_88")) {
|
|
dt = BX_FDD_350ED;
|
|
}
|
|
else if (!strcmp(value, "1_44")) {
|
|
dt = BX_FDD_350HD;
|
|
}
|
|
else if (!strcmp(value, "1_2")) {
|
|
dt = BX_FDD_525HD;
|
|
}
|
|
else if (!strcmp(value, "720k")) {
|
|
dt = BX_FDD_350DD;
|
|
}
|
|
else if (!strcmp(value, "360k")) {
|
|
dt = BX_FDD_525DD;
|
|
}
|
|
else if (!strcmp(value, "none")) {
|
|
dt = BX_FDD_NONE;
|
|
}
|
|
else {
|
|
PARSE_ERR(("%s: %s: unknown type '%s'.", context, params[0], value));
|
|
}
|
|
if (dt >= 0) {
|
|
SIM->get_param_enum("devtype", base)->set(dt);
|
|
}
|
|
}
|
|
else if (!strcmp(params[i], "status=inserted")) {
|
|
SIM->get_param_enum("status", base)->set(BX_INSERTED);
|
|
}
|
|
else if (!strcmp(params[i], "status=ejected")) {
|
|
SIM->get_param_enum("status", base)->set(BX_EJECTED);
|
|
}
|
|
else if (!strncmp(params[i], "write_protected=", 16)) {
|
|
SIM->get_param_bool("readonly", base)->set(atol(¶ms[i][16]));
|
|
}
|
|
else if (!strncmp(params[i], "image=", 6)) {
|
|
/* "image=" means we should get floppy type from image */
|
|
value = params[i] + 6;
|
|
t = get_floppy_type_from_image(value);
|
|
dt = get_floppy_devtype_from_type(t);
|
|
if (t != BX_FLOPPY_UNKNOWN) {
|
|
SIM->get_param_enum("devtype", base)->set(dt);
|
|
SIM->get_param_string("path", base)->set(value);
|
|
SIM->get_param_enum("type", base)->set(t);
|
|
} else
|
|
PARSE_ERR(("%s: %s image size doesn't match one of the supported types.",
|
|
context, params[0]));
|
|
}
|
|
else {
|
|
if (!strncmp(params[i], "2_88=", 5)) {
|
|
t = BX_FLOPPY_2_88;
|
|
value = params[i] + 5;
|
|
}
|
|
else if (!strncmp(params[i], "1_44=", 5)) {
|
|
t = BX_FLOPPY_1_44;
|
|
value = params[i] + 5;
|
|
}
|
|
else if (!strncmp(params[i], "1_2=", 4)) {
|
|
t = BX_FLOPPY_1_2;
|
|
value = params[i] + 4;
|
|
}
|
|
else if (!strncmp(params[i], "720k=", 5)) {
|
|
t = BX_FLOPPY_720K;
|
|
value = params[i] + 5;
|
|
}
|
|
else if (!strncmp(params[i], "360k=", 5)) {
|
|
t = BX_FLOPPY_360K;
|
|
value = params[i] + 5;
|
|
}
|
|
// use CMOS reserved types?
|
|
else if (!strncmp(params[i], "160k=", 5)) {
|
|
t = BX_FLOPPY_160K;
|
|
value = params[i] + 5;
|
|
}
|
|
else if (!strncmp(params[i], "180k=", 5)) {
|
|
t = BX_FLOPPY_180K;
|
|
value = params[i] + 5;
|
|
}
|
|
else if (!strncmp(params[i], "320k=", 5)) {
|
|
t = BX_FLOPPY_320K;
|
|
value = params[i] + 5;
|
|
}
|
|
else {
|
|
t = -1;
|
|
PARSE_ERR(("%s: %s attribute '%s' not understood.", context, params[0],
|
|
params[i]));
|
|
}
|
|
if (t > 0) {
|
|
dt = get_floppy_devtype_from_type(t);
|
|
SIM->get_param_enum("devtype", base)->set(dt);
|
|
SIM->get_param_string("path", base)->set(value);
|
|
SIM->get_param_enum("type", base)->set(t);
|
|
}
|
|
}
|
|
}
|
|
} else if ((!strncmp(params[0], "ata", 3)) && (strlen(params[0]) == 4)) {
|
|
char tmpname[80];
|
|
Bit8u channel = params[0][3];
|
|
|
|
if ((channel < '0') || (channel > '9')) {
|
|
PARSE_ERR(("%s: ataX directive malformed.", context));
|
|
}
|
|
channel-='0';
|
|
if (channel >= BX_MAX_ATA_CHANNEL) {
|
|
PARSE_ERR(("%s: ataX directive malformed.", context));
|
|
}
|
|
|
|
if ((num_params < 2) || (num_params > 5)) {
|
|
PARSE_ERR(("%s: ataX directive malformed.", context));
|
|
}
|
|
sprintf(tmpname, "ata.%d.resources", channel);
|
|
for (i=1; i<num_params; i++) {
|
|
if (bx_parse_param_from_list(context, params[i], (bx_list_c*) SIM->get_param(tmpname)) < 0) {
|
|
PARSE_ERR(("%s: ataX directive malformed.", context));
|
|
}
|
|
}
|
|
} else if ((!strncmp(params[0], "ata", 3)) && (strlen(params[0]) > 4)) {
|
|
// ataX-master, ataX-slave
|
|
Bit8u channel = params[0][3];
|
|
int type = -1;
|
|
Bit32u cylinders = 0, heads = 0, sectors = 0;
|
|
char tmpname[80];
|
|
|
|
if ((channel < '0') || (channel > '9')) {
|
|
PARSE_ERR(("%s: ataX-master/slave directive malformed.", context));
|
|
}
|
|
channel-='0';
|
|
if (channel >= BX_MAX_ATA_CHANNEL) {
|
|
PARSE_ERR(("%s: ataX-master/slave directive malformed.", context));
|
|
}
|
|
|
|
if ((strcmp(¶ms[0][4], "-slave")) &&
|
|
(strcmp(¶ms[0][4], "-master"))) {
|
|
PARSE_ERR(("%s: ataX-master/slave directive malformed.", context));
|
|
}
|
|
|
|
sprintf(tmpname, "ata.%d.%s", channel, ¶ms[0][5]);
|
|
base = (bx_list_c*) SIM->get_param(tmpname);
|
|
for (i=1; i<num_params; i++) {
|
|
if (!strncmp(params[i], "type=", 5)) {
|
|
type = SIM->get_param_enum("type", base)->find_by_name(¶ms[i][5]);
|
|
if (type < 0) {
|
|
PARSE_ERR(("%s: ataX-master/slave: unknown type '%s'", context, ¶ms[i][5]));
|
|
} else {
|
|
SIM->get_param_enum("type", base)->set(type);
|
|
}
|
|
} else if (!strncmp(params[i], "cylinders=", 10)) {
|
|
cylinders = atol(¶ms[i][10]);
|
|
} else if (!strncmp(params[i], "heads=", 6)) {
|
|
heads = atol(¶ms[i][6]);
|
|
} else if (!strncmp(params[i], "spt=", 4)) {
|
|
sectors = atol(¶ms[i][4]);
|
|
} else if (!strcmp(params[i], "translation=echs")) { // synonym of large
|
|
SIM->get_param_enum("translation", base)->set(BX_ATA_TRANSLATION_LARGE);
|
|
} else if (bx_parse_param_from_list(context, params[i], base) < 0) {
|
|
PARSE_ERR(("%s: ataX-master/slave directive malformed.", context));
|
|
}
|
|
}
|
|
|
|
// Check for geometry autodetection mode
|
|
if (type == BX_ATA_DEVICE_DISK) {
|
|
if (strlen(SIM->get_param_string("path", base)->getptr()) > 0) {
|
|
SIM->get_param_num("cylinders", base)->set(cylinders);
|
|
if ((cylinders == 0) && (heads == 0) && (sectors == 0)) {
|
|
PARSE_WARN(("%s: ataX-master/slave CHS set to 0/0/0 - autodetection enabled", context));
|
|
// using heads = 16 and spt = 63 for autodetection (bximage defaults)
|
|
SIM->get_param_num("heads", base)->set(16);
|
|
SIM->get_param_num("spt", base)->set(63);
|
|
} else {
|
|
SIM->get_param_num("heads", base)->set(heads);
|
|
SIM->get_param_num("spt", base)->set(sectors);
|
|
}
|
|
} else {
|
|
SIM->get_param_enum("type", base)->set(BX_ATA_DEVICE_NONE);
|
|
}
|
|
}
|
|
} else if (!strcmp(params[0], "boot")) {
|
|
char tmppath[80];
|
|
if (num_params < 2) {
|
|
PARSE_ERR(("%s: boot directive malformed.", context));
|
|
}
|
|
for (i=1; i<num_params; i++) {
|
|
sprintf(tmppath, "boot_params.boot_drive%d", i);
|
|
if (!strcmp(params[i], "none")) {
|
|
SIM->get_param_enum(tmppath)->set(BX_BOOT_NONE);
|
|
} else if (!strcmp(params[i], "a")) {
|
|
SIM->get_param_enum(tmppath)->set(BX_BOOT_FLOPPYA);
|
|
} else if (!strcmp(params[i], "floppy")) {
|
|
SIM->get_param_enum(tmppath)->set(BX_BOOT_FLOPPYA);
|
|
} else if (!strcmp(params[i], "c")) {
|
|
SIM->get_param_enum(tmppath)->set(BX_BOOT_DISKC);
|
|
} else if (!strcmp(params[i], "disk")) {
|
|
SIM->get_param_enum(tmppath)->set(BX_BOOT_DISKC);
|
|
} else if (!strcmp(params[i], "cdrom")) {
|
|
SIM->get_param_enum(tmppath)->set(BX_BOOT_CDROM);
|
|
} else if (!strcmp(params[i], "network")) {
|
|
SIM->get_param_enum(tmppath)->set(BX_BOOT_NETWORK);
|
|
} else {
|
|
PARSE_ERR(("%s: boot directive with unknown boot drive '%s'. use 'floppy', 'disk', 'cdrom' or 'network'.", context, params[i]));
|
|
}
|
|
}
|
|
if (SIM->get_param_enum(BXPN_BOOTDRIVE1)->get() == BX_BOOT_NONE) {
|
|
PARSE_ERR(("%s: first boot drive must be one of 'floppy', 'disk' or 'cdrom'.", context));
|
|
}
|
|
if ((SIM->get_param_enum(BXPN_BOOTDRIVE1)->get() == SIM->get_param_enum(BXPN_BOOTDRIVE2)->get()) ||
|
|
(SIM->get_param_enum(BXPN_BOOTDRIVE1)->get() == SIM->get_param_enum(BXPN_BOOTDRIVE3)->get()) ||
|
|
((SIM->get_param_enum(BXPN_BOOTDRIVE3)->get() != BX_BOOT_NONE) &&
|
|
(SIM->get_param_enum(BXPN_BOOTDRIVE2)->get() == SIM->get_param_enum(BXPN_BOOTDRIVE3)->get()))) {
|
|
PARSE_ERR(("%s: a boot drive appears twice in boot sequence.", context));
|
|
}
|
|
} else if (!strcmp(params[0], "floppy_bootsig_check")) {
|
|
if (num_params != 2) {
|
|
PARSE_ERR(("%s: floppy_bootsig_check directive malformed.", context));
|
|
}
|
|
if (strncmp(params[1], "disabled=", 9)) {
|
|
PARSE_ERR(("%s: floppy_bootsig_check directive malformed.", context));
|
|
}
|
|
if (parse_param_bool(params[1], 9, BXPN_FLOPPYSIGCHECK) < 0) {
|
|
PARSE_ERR(("%s: floppy_bootsig_check directive malformed.", context));
|
|
}
|
|
} else if (!strcmp(params[0], "log")) {
|
|
if (num_params != 2) {
|
|
PARSE_ERR(("%s: log directive has wrong # args.", context));
|
|
}
|
|
SIM->get_param_string(BXPN_LOG_FILENAME)->set(params[1]);
|
|
} else if (!strcmp(params[0], "logprefix")) {
|
|
if (num_params != 2) {
|
|
PARSE_ERR(("%s: logprefix directive has wrong # args.", context));
|
|
}
|
|
SIM->get_param_string(BXPN_LOG_PREFIX)->set(params[1]);
|
|
} else if (!strcmp(params[0], "debugger_log")) {
|
|
if (num_params != 2) {
|
|
PARSE_ERR(("%s: debugger_log directive has wrong # args.", context));
|
|
}
|
|
SIM->get_param_string(BXPN_DEBUGGER_LOG_FILENAME)->set(params[1]);
|
|
} else if (!strcmp(params[0], "panic")) {
|
|
if (num_params < 2) {
|
|
PARSE_ERR(("%s: panic directive malformed.", context));
|
|
}
|
|
if (parse_log_options(context, num_params, params) < 0) {
|
|
return -1;
|
|
}
|
|
} else if (!strcmp(params[0], "error")) {
|
|
if (num_params < 2) {
|
|
PARSE_ERR(("%s: error directive malformed.", context));
|
|
}
|
|
if (parse_log_options(context, num_params, params) < 0) {
|
|
return -1;
|
|
}
|
|
} else if (!strcmp(params[0], "info")) {
|
|
if (num_params < 2) {
|
|
PARSE_ERR(("%s: info directive malformed.", context));
|
|
}
|
|
if (parse_log_options(context, num_params, params) < 0) {
|
|
return -1;
|
|
}
|
|
} else if (!strcmp(params[0], "debug")) {
|
|
if (num_params < 2) {
|
|
PARSE_ERR(("%s: debug directive malformed.", context));
|
|
}
|
|
if (parse_log_options(context, num_params, params) < 0) {
|
|
return -1;
|
|
}
|
|
} else if (!strcmp(params[0], "cpu")) {
|
|
if (num_params < 2) {
|
|
PARSE_ERR(("%s: cpu directive malformed.", context));
|
|
}
|
|
for (i=1; i<num_params; i++) {
|
|
if (!strncmp(params[i], "count=", 6)) {
|
|
unsigned processors = 1, cores = 1, threads = 1;
|
|
sscanf(¶ms[i][6], "%u:%u:%u", &processors, &cores, &threads);
|
|
unsigned smp_threads = cores*threads*processors;
|
|
if (smp_threads < 1) {
|
|
PARSE_ERR(("%s: at least one CPU thread should be defined, cpu directive malformed", context));
|
|
}
|
|
SIM->get_param_num(BXPN_CPU_NPROCESSORS)->set(processors);
|
|
SIM->get_param_num(BXPN_CPU_NCORES)->set(cores);
|
|
SIM->get_param_num(BXPN_CPU_NTHREADS)->set(threads);
|
|
} else if (bx_parse_param_from_list(context, params[i], (bx_list_c*) SIM->get_param("cpu")) < 0) {
|
|
PARSE_ERR(("%s: cpu directive malformed.", context));
|
|
}
|
|
}
|
|
#if BX_CPU_LEVEL >= 4
|
|
} else if (!strcmp(params[0], "cpuid")) {
|
|
if (num_params < 2) {
|
|
PARSE_ERR(("%s: cpuid directive malformed.", context));
|
|
}
|
|
for (i=1; i<num_params; i++) {
|
|
if (bx_parse_param_from_list(context, params[i], (bx_list_c*) SIM->get_param("cpuid")) < 0) {
|
|
PARSE_ERR(("%s: cpuid directive malformed.", context));
|
|
}
|
|
}
|
|
#endif
|
|
} else if (!strcmp(params[0], "megs")) {
|
|
if (num_params != 2) {
|
|
PARSE_ERR(("%s: megs directive: wrong # args.", context));
|
|
}
|
|
SIM->get_param_num(BXPN_MEM_SIZE)->set(atol(params[1]));
|
|
SIM->get_param_num(BXPN_HOST_MEM_SIZE)->set(atol(params[1]));
|
|
} else if (!strcmp(params[0], "memory")) {
|
|
if (num_params < 2) {
|
|
PARSE_ERR(("%s: memory directive malformed.", context));
|
|
}
|
|
for (i=1; i<num_params; i++) {
|
|
if (bx_parse_param_from_list(context, params[i], (bx_list_c*) SIM->get_param(BXPN_MEMORY)) < 0) {
|
|
PARSE_ERR(("%s: memory directive malformed.", context));
|
|
}
|
|
}
|
|
} else if (!strcmp(params[0], "romimage")) {
|
|
if ((num_params < 2) || (num_params > 4)) {
|
|
PARSE_ERR(("%s: romimage directive: wrong # args.", context));
|
|
}
|
|
// set to default value 0 (auto-detect if no specified)
|
|
SIM->get_param_num(BXPN_ROM_ADDRESS)->set(0);
|
|
for (i=1; i<num_params; i++) {
|
|
if (!strncmp(params[i], "file=", 5)) {
|
|
SIM->get_param_string(BXPN_ROM_PATH)->set(¶ms[i][5]);
|
|
} else if (!strncmp(params[i], "address=", 8)) {
|
|
if ((params[i][8] == '0') && (params[i][9] == 'x'))
|
|
SIM->get_param_num(BXPN_ROM_ADDRESS)->set(strtoul(¶ms[i][8], NULL, 16));
|
|
else
|
|
SIM->get_param_num(BXPN_ROM_ADDRESS)->set(strtoul(¶ms[i][8], NULL, 10));
|
|
} else if (!strncmp(params[i], "options=", 8)) {
|
|
SIM->get_param_string(BXPN_ROM_OPTIONS)->set(¶ms[i][8]);
|
|
} else {
|
|
PARSE_ERR(("%s: romimage directive malformed.", context));
|
|
}
|
|
}
|
|
} else if (!strcmp(params[0], "vgaromimage")) {
|
|
if (num_params != 2) {
|
|
PARSE_ERR(("%s: vgaromimage directive: wrong # args.", context));
|
|
}
|
|
if (!strncmp(params[1], "file=", 5)) {
|
|
SIM->get_param_string(BXPN_VGA_ROM_PATH)->set(¶ms[1][5]);
|
|
} else {
|
|
PARSE_ERR(("%s: vgaromimage directive malformed.", context));
|
|
}
|
|
} else if (!strncmp(params[0], "optromimage", 11)) {
|
|
int num = atoi(¶ms[0][11]);
|
|
char pname[16];
|
|
if ((num < 1) || (num > BX_N_OPTROM_IMAGES)) {
|
|
PARSE_ERR(("%s: optromimage%d: not supported", context, num));
|
|
}
|
|
if (num_params > 3) {
|
|
PARSE_ERR(("%s: optromimage%d directive: wrong # args.", context, num));
|
|
}
|
|
sprintf(pname, "%s.%d", BXPN_OPTROM_BASE, num);
|
|
base = (bx_list_c*) SIM->get_param(pname);
|
|
for (i=1; i<num_params; i++) {
|
|
if (!strncmp(params[i], "file=", 5)) {
|
|
SIM->get_param_string("file", base)->set(¶ms[i][5]);
|
|
} else if (!strncmp(params[i], "address=", 8)) {
|
|
if ((params[i][8] == '0') && (params[2][9] == 'x'))
|
|
SIM->get_param_num("address", base)->set(strtoul(¶ms[i][8], NULL, 16));
|
|
else
|
|
SIM->get_param_num("address", base)->set(strtoul(¶ms[i][8], NULL, 10));
|
|
} else {
|
|
PARSE_ERR(("%s: optromimage%d directive malformed.", context, num));
|
|
}
|
|
}
|
|
} else if (!strncmp(params[0], "optramimage", 11)) {
|
|
int num = atoi(¶ms[0][11]);
|
|
char pname[16];
|
|
if ((num < 1) || (num > BX_N_OPTRAM_IMAGES)) {
|
|
PARSE_ERR(("%s: optramimage%d: not supported", context, num));
|
|
}
|
|
if (num_params > 3) {
|
|
PARSE_ERR(("%s: optramimage%d directive: wrong # args.", context, num));
|
|
}
|
|
sprintf(pname, "%s.%d", BXPN_OPTRAM_BASE, num);
|
|
base = (bx_list_c*) SIM->get_param(pname);
|
|
for (i=1; i<num_params; i++) {
|
|
if (!strncmp(params[i], "file=", 5)) {
|
|
SIM->get_param_string("file", base)->set(¶ms[i][5]);
|
|
} else if (!strncmp(params[i], "address=", 8)) {
|
|
if ((params[i][8] == '0') && (params[2][9] == 'x'))
|
|
SIM->get_param_num("address", base)->set(strtoul(¶ms[i][8], NULL, 16));
|
|
else
|
|
SIM->get_param_num("address", base)->set(strtoul(¶ms[i][8], NULL, 10));
|
|
} else {
|
|
PARSE_ERR(("%s: optramimage%d directive malformed.", context, num));
|
|
}
|
|
}
|
|
} else if (!strcmp(params[0], "vga")) {
|
|
if (num_params < 2) {
|
|
PARSE_ERR(("%s: vga directive malformed.", context));
|
|
}
|
|
for (i=1; i<num_params; i++) {
|
|
if (!strncmp(params[i], "extension=", 10)) {
|
|
SIM->get_param_enum(BXPN_VGA_EXTENSION)->set_by_name(¶ms[i][10]);
|
|
} else if (!strncmp(params[i], "update_freq=", 12)) {
|
|
SIM->get_param_num(BXPN_VGA_UPDATE_FREQUENCY)->set(atol(¶ms[i][12]));
|
|
} else if (!strncmp(params[i], "realtime=", 9)) {
|
|
SIM->get_param_bool(BXPN_VGA_REALTIME)->set(atol(¶ms[i][9]));
|
|
} else if (!strncmp(params[i], "ddc=", 4)) {
|
|
const char *strval = ¶ms[i][4];
|
|
if (strncmp(strval, "file:", 5)) {
|
|
SIM->get_param_enum(BXPN_DDC_MODE)->set_by_name(strval);
|
|
} else {
|
|
SIM->get_param_enum(BXPN_DDC_MODE)->set(BX_DDC_MODE_FILE);
|
|
SIM->get_param_string(BXPN_DDC_FILE)->set(strval+5);
|
|
}
|
|
} else {
|
|
PARSE_ERR(("%s: vga directive malformed.", context));
|
|
}
|
|
}
|
|
} else if (!strcmp(params[0], "keyboard")) {
|
|
if (num_params < 2) {
|
|
PARSE_ERR(("%s: keyboard directive malformed.", context));
|
|
}
|
|
for (i=1; i<num_params; i++) {
|
|
if (!strncmp(params[i], "keymap=", 7)) {
|
|
char *kmap = ¶ms[i][7];
|
|
SIM->get_param_bool(BXPN_KBD_USEMAPPING)->set((strlen(kmap) > 0) && (strcmp(kmap, "none")));
|
|
SIM->get_param_string(BXPN_KBD_KEYMAP)->set(kmap);
|
|
} else if (bx_parse_param_from_list(context, params[i], (bx_list_c*) SIM->get_param(BXPN_KEYBOARD)) < 0) {
|
|
PARSE_ERR(("%s: keyboard directive malformed.", context));
|
|
}
|
|
}
|
|
} else if (!strcmp(params[0], "mouse")) {
|
|
if (num_params < 2) {
|
|
PARSE_ERR(("%s: mouse directive malformed.", context));
|
|
}
|
|
for (i=1; i<num_params; i++) {
|
|
if (bx_parse_param_from_list(context, params[i], (bx_list_c*) SIM->get_param(BXPN_MOUSE)) < 0) {
|
|
PARSE_ERR(("%s: mouse directive malformed.", context));
|
|
}
|
|
}
|
|
} else if (!strcmp(params[0], "private_colormap")) {
|
|
if (num_params != 2) {
|
|
PARSE_ERR(("%s: private_colormap directive malformed.", context));
|
|
}
|
|
if (strncmp(params[1], "enabled=", 8)) {
|
|
PARSE_ERR(("%s: private_colormap directive malformed.", context));
|
|
}
|
|
if (parse_param_bool(params[1], 8, BXPN_PRIVATE_COLORMAP) < 0) {
|
|
PARSE_ERR(("%s: private_colormap directive malformed.", context));
|
|
}
|
|
} else if (!strcmp(params[0], "fullscreen")) {
|
|
#if BX_WITH_AMIGAOS
|
|
if (num_params != 2) {
|
|
PARSE_ERR(("%s: fullscreen directive malformed.", context));
|
|
}
|
|
if (strncmp(params[1], "enabled=", 8)) {
|
|
PARSE_ERR(("%s: fullscreen directive malformed.", context));
|
|
}
|
|
if (parse_param_bool(params[1], 8, BXPN_FULLSCREEN) < 0) {
|
|
PARSE_ERR(("%s: fullscreen directive malformed.", context));
|
|
}
|
|
#endif
|
|
} else if (!strcmp(params[0], "screenmode")) {
|
|
#if BX_WITH_AMIGAOS
|
|
if (num_params != 2) {
|
|
PARSE_ERR(("%s: screenmode directive malformed.", context));
|
|
}
|
|
if (strncmp(params[1], "name=", 5)) {
|
|
PARSE_ERR(("%s: screenmode directive malformed.", context));
|
|
}
|
|
SIM->get_param_string(BXPN_SCREENMODE)->set(¶ms[1][5]);
|
|
#endif
|
|
} else if (!strcmp(params[0], "pci")) {
|
|
char tmpdev[80];
|
|
int enabled = -1;
|
|
bool chipset = 0;
|
|
for (i=1; i<num_params; i++) {
|
|
if (!strncmp(params[i], "enabled=", 8)) {
|
|
enabled = atol(¶ms[i][8]);
|
|
} else if (!strncmp(params[i], "chipset=", 8)) {
|
|
if (!SIM->get_param_enum(BXPN_PCI_CHIPSET)->set_by_name(¶ms[i][8])) {
|
|
PARSE_ERR(("%s: pci: unknown chipset '%s'", context, ¶ms[i][8]));
|
|
} else {
|
|
chipset = 1;
|
|
}
|
|
} else if ((!strncmp(params[i], "slot", 4)) && (params[i][5] == '=')) {
|
|
slot = atol(¶ms[i][4]);
|
|
if ((slot > 0) && (slot < 6)) {
|
|
sprintf(tmpdev, "pci.slot.%d", slot);
|
|
if (strlen(¶ms[i][6]) > 0) {
|
|
if (!SIM->get_param_enum(tmpdev)->set_by_name(¶ms[i][6])) {
|
|
PARSE_ERR(("%s: unknown plugin '%s' at PCI slot #%d.",
|
|
context, ¶ms[i][6], slot));
|
|
}
|
|
} else {
|
|
SIM->get_param_enum(tmpdev)->set_by_name("none");
|
|
}
|
|
} else {
|
|
PARSE_ERR(("%s: unknown PCI slot number #%d.", context, slot));
|
|
}
|
|
} else if (!strncmp(params[i], "advopts=", 8)) {
|
|
SIM->get_param_string(BXPN_PCI_ADV_OPTS)->set(¶ms[i][8]);
|
|
} else {
|
|
PARSE_ERR(("%s: pci: unknown parameter '%s'.", context, params[i]));
|
|
}
|
|
}
|
|
if (enabled == 0) {
|
|
SIM->get_param_bool(BXPN_PCI_ENABLED)->set(0);
|
|
} else if (enabled == 1) {
|
|
if (chipset == 1) {
|
|
SIM->get_param_bool(BXPN_PCI_ENABLED)->set(1);
|
|
} else {
|
|
PARSE_ERR(("%s: pci: chipset not specified", context));
|
|
}
|
|
}
|
|
} else if (!strcmp(params[0], "cmosimage")) {
|
|
for (i=1; i<num_params; i++) {
|
|
if (!strncmp(params[i], "file=", 5)) {
|
|
SIM->get_param_string(BXPN_CMOSIMAGE_PATH)->set(¶ms[i][5]);
|
|
} else if (!strncmp(params[i], "rtc_init=time0",14)) {
|
|
SIM->get_param_bool(BXPN_CMOSIMAGE_RTC_INIT)->set(0);
|
|
} else if (!strncmp(params[i], "rtc_init=image",14)) {
|
|
SIM->get_param_bool(BXPN_CMOSIMAGE_RTC_INIT)->set(1);
|
|
} else {
|
|
BX_ERROR(("%s: unknown parameter for cmosimage ignored.", context));
|
|
}
|
|
}
|
|
if (strlen(SIM->get_param_string(BXPN_CMOSIMAGE_PATH)->getptr()) > 0) {
|
|
SIM->get_param_bool(BXPN_CMOSIMAGE_ENABLED)->set(1);
|
|
}
|
|
} else if (!strcmp(params[0], "clock")) {
|
|
const char months[] = "Jan Feb Mar Apr May Jun Jul Aug Sep Oct Nov Dec ";
|
|
char wday[4], mon[4];
|
|
int n, year;
|
|
struct tm tm_time;
|
|
for (i=1; i<num_params; i++) {
|
|
if (!strncmp(params[i], "sync=", 5)) {
|
|
SIM->get_param_enum(BXPN_CLOCK_SYNC)->set_by_name(¶ms[i][5]);
|
|
}
|
|
else if (!strncmp(params[i], "rtc_sync=", 9)) {
|
|
SIM->get_param_bool(BXPN_CLOCK_RTC_SYNC)->set(atol(¶ms[i][9]));
|
|
}
|
|
else if (!strcmp(params[i], "time0=local")) {
|
|
SIM->get_param_num(BXPN_CLOCK_TIME0)->set(BX_CLOCK_TIME0_LOCAL);
|
|
}
|
|
else if (!strcmp(params[i], "time0=utc")) {
|
|
SIM->get_param_num(BXPN_CLOCK_TIME0)->set(BX_CLOCK_TIME0_UTC);
|
|
}
|
|
else if (!strncmp(params[i], "time0=", 6)) {
|
|
if (isalpha(params[i][6])) {
|
|
memset(&tm_time, 0, sizeof(tm_time));
|
|
n = sscanf(¶ms[i][6], "%3s %3s %2d %2d:%2d:%2d %d", wday, mon, &tm_time.tm_mday,
|
|
&tm_time.tm_hour, &tm_time.tm_min, &tm_time.tm_sec, &year);
|
|
if ((n == 7) && (strstr(months, mon) != NULL)) {
|
|
tm_time.tm_year = year - 1900;
|
|
tm_time.tm_mon = 12 - ((int)strlen(strstr(months, mon)) / 4);
|
|
tm_time.tm_isdst = -1;
|
|
SIM->get_param_num(BXPN_CLOCK_TIME0)->set(mktime(&tm_time));
|
|
} else {
|
|
PARSE_ERR(("%s: time0 string format malformed.", context));
|
|
}
|
|
} else {
|
|
Bit64s tmptm=atol(¶ms[i][6]);
|
|
SIM->get_param_num(BXPN_CLOCK_TIME0)->set(tmptm);
|
|
}
|
|
}
|
|
else {
|
|
BX_ERROR(("%s: unknown parameter for clock ignored.", context));
|
|
}
|
|
}
|
|
} else if (!strcmp(params[0], "sound")) {
|
|
#if BX_SUPPORT_SOUNDLOW
|
|
static const char default_drv[] = BX_SOUND_LOWLEVEL_NAME;
|
|
const char *driver;
|
|
for (i=1; i<num_params; i++) {
|
|
if (!strncmp(params[i], "driver=", 7)) {
|
|
if (!strcmp(¶ms[i][7], "default")) {
|
|
driver = default_drv;
|
|
} else {
|
|
driver = ¶ms[i][7];
|
|
}
|
|
SIM->get_param_enum(BXPN_SOUND_WAVEOUT_DRV)->set_by_name(driver);
|
|
SIM->get_param_enum(BXPN_SOUND_WAVEIN_DRV)->set_by_name(driver);
|
|
SIM->get_param_enum(BXPN_SOUND_MIDIOUT_DRV)->set_by_name(driver);
|
|
} else if (bx_parse_param_from_list(context, params[i], (bx_list_c*) SIM->get_param(BXPN_SOUNDLOW)) < 0) {
|
|
BX_ERROR(("%s: unknown parameter for sound ignored.", context));
|
|
}
|
|
}
|
|
#else
|
|
PARSE_ERR(("%s: Bochs is not compiled with lowlevel sound support", context));
|
|
#endif
|
|
} else if (!strcmp(params[0], "gdbstub")) {
|
|
#if BX_GDBSTUB
|
|
if (num_params < 2) {
|
|
PARSE_ERR(("%s: gdbstub directive: wrong # args.", context));
|
|
}
|
|
base = (bx_list_c*) SIM->get_param(BXPN_GDBSTUB);
|
|
for (i=1; i<num_params; i++) {
|
|
if (!strncmp(params[i], "enabled=", 8)) {
|
|
if (params[i][8] == '0') {
|
|
SIM->get_param_bool("enabled", base)->set(0);
|
|
BX_INFO(("Disabled gdbstub"));
|
|
bx_dbg.gdbstub_enabled = 0;
|
|
}
|
|
else if (params[i][8] == '1') {
|
|
SIM->get_param_bool("enabled", base)->set(1);
|
|
BX_INFO(("Enabled gdbstub"));
|
|
bx_dbg.gdbstub_enabled = 1;
|
|
}
|
|
else {
|
|
PARSE_ERR(("%s: gdbstub directive malformed.", context));
|
|
}
|
|
}
|
|
else if (!strncmp(params[i], "port=", 5)) {
|
|
SIM->get_param_num("port", base)->set(atoi(¶ms[i][5]));
|
|
}
|
|
else if (!strncmp(params[i], "text_base=", 10)) {
|
|
SIM->get_param_num("text_base", base)->set(atoi(¶ms[i][10]));
|
|
}
|
|
else if (!strncmp(params[i], "data_base=", 10)) {
|
|
SIM->get_param_num("data_base", base)->set(atoi(¶ms[i][10]));
|
|
}
|
|
else if (!strncmp(params[i], "bss_base=", 9)) {
|
|
SIM->get_param_num("bss_base", base)->set(atoi(¶ms[i][9]));
|
|
}
|
|
else {
|
|
PARSE_ERR(("%s: gdbstub directive malformed.", context));
|
|
}
|
|
}
|
|
#else
|
|
PARSE_ERR(("%s: Bochs is not compiled with gdbstub support", context));
|
|
#endif
|
|
} else if (!strcmp(params[0], "magic_break")) {
|
|
#if BX_DEBUGGER
|
|
bx_dbg.magic_break = 0;
|
|
if (num_params != 2) {
|
|
PARSE_ERR(("%s: magic_break directive: wrong # args.", context));
|
|
}
|
|
if (strncmp(params[1], "enabled=", 8)) {
|
|
PARSE_ERR(("%s: magic_break directive malformed.", context));
|
|
}
|
|
if (params[1][8] == '0') {
|
|
BX_INFO(("Ignoring magic break points"));
|
|
} else if (params[1][8] == '1') {
|
|
bx_dbg_set_magic_bp_mask(bx_dbg_get_magic_bp_mask_from_str(params[1]));
|
|
if (0 == bx_dbg.magic_break) {
|
|
// bx if not specified for backward compatibility
|
|
bx_dbg_set_magic_bp_mask(bx_dbg_get_magic_bp_mask_from_str("bx"));
|
|
}
|
|
} else {
|
|
PARSE_ERR(("%s: magic_break directive malformed", context));
|
|
}
|
|
#else
|
|
PARSE_WARN(("%s: Bochs is not compiled with internal debugger support", context));
|
|
#endif
|
|
} else if (!strcmp(params[0], "debug_symbols")) {
|
|
if (parse_debug_symbols(context, (const char **)(params + 1), num_params - 1) < 0) {
|
|
return -1;
|
|
}
|
|
} else if (!strcmp(params[0], "print_timestamps")) {
|
|
if (num_params != 2) {
|
|
PARSE_ERR(("%s: print_timestamps directive: wrong # args.", context));
|
|
}
|
|
if (strncmp(params[1], "enabled=", 8)) {
|
|
PARSE_ERR(("%s: print_timestamps directive malformed.", context));
|
|
}
|
|
if (params[1][8] == '0' || params[1][8] == '1') {
|
|
bx_dbg.print_timestamps = params[1][8] - '0';
|
|
}
|
|
else {
|
|
PARSE_ERR(("%s: print_timestamps directive malformed.", context));
|
|
}
|
|
} else if (!strcmp(params[0], "port_e9_hack")) {
|
|
for (i=1; i<num_params; i++) {
|
|
if (bx_parse_param_from_list(context, params[i], (bx_list_c*) SIM->get_param(BXPN_PORT_E9_HACK_ROOT)) < 0) {
|
|
PARSE_ERR(("%s: port_e9_hack directive malformed.", context));
|
|
}
|
|
}
|
|
} else if (!strcmp(params[0], "iodebug")) {
|
|
#if BX_SUPPORT_IODEBUG
|
|
if (num_params != 2) {
|
|
PARSE_ERR(("%s: iodebug directive: wrong # args.", context));
|
|
}
|
|
if (!strncmp(params[1], "all_rings=", 10)) {
|
|
if (parse_param_bool(params[1], 10, BXPN_IODEBUG_ALL_RINGS) < 0) {
|
|
PARSE_ERR(("%s: all_rings option malformed.", context));
|
|
}
|
|
} else {
|
|
PARSE_ERR(("%s: iodebug: invalid parameter %s", context, params[1]));
|
|
}
|
|
#else
|
|
PARSE_WARN(("%s: Bochs is not compiled with iodebug support", context));
|
|
#endif
|
|
} else if (!strcmp(params[0], "load32bitOSImage")) {
|
|
PARSE_ERR(("%s: load32bitOSImage: This legacy feature is no longer supported.", context));
|
|
} else if (SIM->is_addon_option(params[0])) {
|
|
// add-on options handled by registered functions
|
|
return SIM->parse_addon_option(context, num_params, ¶ms[0]);
|
|
} else if (is_deprecated_option(params[0], &newparam)) {
|
|
PARSE_ERR(("%s: '%s' is deprecated - use '%s' option instead.", context, params[0], newparam));
|
|
} else if (bx_opt_plugin_available(params[0])) {
|
|
// treat unknown option as plugin name and try to load it
|
|
if (SIM->opt_plugin_ctrl(params[0], 1)) {
|
|
if (SIM->is_addon_option(params[0])) {
|
|
// after loading the plugin a bochsrc option with it's name must exist
|
|
return SIM->parse_addon_option(context, num_params, ¶ms[0]);
|
|
} else {
|
|
PARSE_ERR(("%s: directive '%s' not understood", context, params[0]));
|
|
}
|
|
}
|
|
} else {
|
|
PARSE_ERR(("%s: directive '%s' not understood", context, params[0]));
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
|
|
int bx_write_param_list(FILE *fp, bx_list_c *base, const char *optname, bool multiline)
|
|
{
|
|
char bxrcline[BX_PATHNAME_LEN], tmpstr[BX_PATHNAME_LEN];
|
|
bool newline = 1;
|
|
int p = 0;
|
|
|
|
if (base == NULL) return -1;
|
|
if (!base->get_enabled()) return -1;
|
|
bxrcline[0] = 0;
|
|
for (int i = 0; i < base->get_size(); i++) {
|
|
if (newline) {
|
|
if (strlen(bxrcline) > 0) {
|
|
fprintf(fp, "%s\n", bxrcline);
|
|
}
|
|
if (optname == NULL) {
|
|
sprintf(bxrcline, "%s: ", base->get_name());
|
|
} else if (isspace(optname[strlen(optname)-1])) {
|
|
sprintf(bxrcline, "%s", optname);
|
|
} else {
|
|
sprintf(bxrcline, "%s: ", optname);
|
|
}
|
|
newline = 0;
|
|
p = 0;
|
|
}
|
|
bx_param_c *param = base->get(i);
|
|
if (param->get_enabled() && ((param->get_options() & param->CI_ONLY) == 0)) {
|
|
if (p > 0) {
|
|
strcat(bxrcline, ", ");
|
|
}
|
|
sprintf(tmpstr, "%s=", param->get_name());
|
|
strcat(bxrcline, tmpstr);
|
|
switch (param->get_type()) {
|
|
case BXT_PARAM_NUM:
|
|
case BXT_PARAM_BOOL:
|
|
case BXT_PARAM_ENUM:
|
|
case BXT_PARAM_STRING:
|
|
case BXT_PARAM_BYTESTRING:
|
|
param->dump_param(tmpstr, BX_PATHNAME_LEN, 1);
|
|
break;
|
|
default:
|
|
BX_ERROR(("bx_write_param_list(): unsupported parameter type"));
|
|
tmpstr[0] = 0;
|
|
}
|
|
strcat(bxrcline, tmpstr);
|
|
p++;
|
|
}
|
|
if (multiline && (strlen(bxrcline) > 80)) {
|
|
newline = 1;
|
|
}
|
|
}
|
|
fprintf(fp, "%s\n", bxrcline);
|
|
return 0;
|
|
}
|
|
|
|
static const char *fdtypes[] = {
|
|
"none", "1_2", "1_44", "2_88", "720k", "360k", "160k", "180k", "320k"
|
|
};
|
|
|
|
int bx_write_floppy_options(FILE *fp, int drive)
|
|
{
|
|
char devtype[80], path[80], type[80], status[80], readonly[80];
|
|
int ftype;
|
|
|
|
BX_ASSERT(drive==0 || drive==1);
|
|
sprintf(devtype, "floppy.%d.devtype", drive);
|
|
sprintf(path, "floppy.%d.path", drive);
|
|
sprintf(type, "floppy.%d.type", drive);
|
|
sprintf(status, "floppy.%d.status", drive);
|
|
sprintf(readonly, "floppy.%d.readonly", drive);
|
|
ftype = SIM->get_param_enum(devtype)->get();
|
|
if (ftype == BX_FDD_NONE) {
|
|
fprintf(fp, "# no floppy%c\n", (char)'a'+drive);
|
|
return 0;
|
|
} else {
|
|
fprintf(fp, "floppy%c: type=", (char)'a'+drive);
|
|
if (ftype == BX_FDD_350ED) {
|
|
fprintf(fp, "2_88");
|
|
} else if (ftype == BX_FDD_350HD) {
|
|
fprintf(fp, "1_44");
|
|
} else if (ftype == BX_FDD_525HD) {
|
|
fprintf(fp, "1_2");
|
|
} else if (ftype == BX_FDD_350DD) {
|
|
fprintf(fp, "720k");
|
|
} else if (ftype == BX_FDD_525DD) {
|
|
fprintf(fp, "360k");
|
|
}
|
|
}
|
|
if ((SIM->get_param_enum(type)->get() > BX_FLOPPY_NONE) &&
|
|
(SIM->get_param_enum(type)->get() <= BX_FLOPPY_LAST)) {
|
|
fprintf(fp, ", %s=\"%s\", status=%s, write_protected=%d",
|
|
fdtypes[SIM->get_param_enum(type)->get() - BX_FLOPPY_NONE],
|
|
SIM->get_param_string(path)->getptr(),
|
|
SIM->get_param_enum(status)->get_selected(),
|
|
SIM->get_param_bool(readonly)->get());
|
|
}
|
|
fprintf(fp, "\n");
|
|
return 0;
|
|
}
|
|
|
|
#if BX_SUPPORT_PCIUSB
|
|
int bx_write_usb_options(FILE *fp, int maxports, bx_list_c *base)
|
|
{
|
|
char tmpname[24], tmpstr[BX_PATHNAME_LEN];
|
|
|
|
fprintf(fp, "usb_%s: enabled=%d", base->get_name(), SIM->get_param_bool("enabled", base)->get());
|
|
|
|
// if we are the ehci, we need to add the companion= parameter
|
|
if (base == SIM->get_param(BXPN_USB_EHCI))
|
|
fprintf(fp, ", companion=%s", SIM->get_param_enum(BXPN_EHCI_COMPANION)->get_selected());
|
|
|
|
// if we are the xhci, we need to add the model= and n_ports= parameters
|
|
if (base == SIM->get_param(BXPN_USB_XHCI)) {
|
|
fprintf(fp, ", model=%s", SIM->get_param_enum(BXPN_XHCI_MODEL)->get_selected());
|
|
fprintf(fp, ", n_ports=%i", SIM->get_param_num(BXPN_XHCI_N_PORTS)->get());
|
|
}
|
|
|
|
if (SIM->get_param_bool("enabled", base)->get()) {
|
|
for (int i = 1; i <= maxports; i++) {
|
|
sprintf(tmpname, "port%d.device", i);
|
|
SIM->get_param_enum(tmpname, base)->dump_param(tmpstr, BX_PATHNAME_LEN, 1);
|
|
fprintf(fp, ", port%d=%s", i, tmpstr);
|
|
sprintf(tmpname, "port%d.options", i);
|
|
SIM->get_param_string(tmpname, base)->dump_param(tmpstr, BX_PATHNAME_LEN, 1);
|
|
fprintf(fp, ", options%d=%s", i, tmpstr);
|
|
}
|
|
}
|
|
fprintf(fp, "\n");
|
|
return 0;
|
|
}
|
|
#endif
|
|
|
|
int bx_write_clock_cmos_options(FILE *fp)
|
|
{
|
|
fprintf(fp, "clock: sync=%s", SIM->get_param_enum(BXPN_CLOCK_SYNC)->get_selected());
|
|
|
|
switch (SIM->get_param_num(BXPN_CLOCK_TIME0)->get64()) {
|
|
case BX_CLOCK_TIME0_LOCAL:
|
|
fprintf(fp, ", time0=local");
|
|
break;
|
|
case BX_CLOCK_TIME0_UTC:
|
|
fprintf(fp, ", time0=utc");
|
|
break;
|
|
default:
|
|
fprintf(fp, ", time0=" FMT_LL "d", SIM->get_param_num(BXPN_CLOCK_TIME0)->get64());
|
|
}
|
|
|
|
fprintf(fp, ", rtc_sync=%d\n", SIM->get_param_bool(BXPN_CLOCK_RTC_SYNC)->get());
|
|
|
|
if (strlen(SIM->get_param_string(BXPN_CMOSIMAGE_PATH)->getptr()) > 0) {
|
|
fprintf(fp, "cmosimage: file=%s, ", SIM->get_param_string(BXPN_CMOSIMAGE_PATH)->getptr());
|
|
fprintf(fp, "rtc_init=%s\n", SIM->get_param_bool(BXPN_CMOSIMAGE_RTC_INIT)->get()?"image":"time0");
|
|
} else {
|
|
fprintf(fp, "# no cmosimage\n");
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
int bx_write_log_options(FILE *fp, bx_list_c *base)
|
|
{
|
|
char pname[20];
|
|
bx_list_c *logfn, *loglev;
|
|
bx_param_num_c *mparam;
|
|
int action, def_action, level, mod;
|
|
|
|
fprintf(fp, "log: %s\n", SIM->get_param_string("filename", base)->getptr());
|
|
fprintf(fp, "logprefix: %s\n", SIM->get_param_string("prefix", base)->getptr());
|
|
|
|
strcpy(pname, "general.logfn");
|
|
logfn = (bx_list_c*) SIM->get_param(pname);
|
|
for (level = 0; level < N_LOGLEV; level++) {
|
|
loglev = (bx_list_c*) logfn->get(level);
|
|
def_action = SIM->get_default_log_action(level);
|
|
fprintf(fp, "%s: action=%s", loglev->get_name(), SIM->get_action_name(def_action));
|
|
// stage #1: save log actions of existing modules
|
|
for (mod = 0; mod < SIM->get_n_log_modules(); mod++) {
|
|
action = SIM->get_log_action(mod, level);
|
|
if (action != def_action) {
|
|
fprintf(fp, ", %s=%s", SIM->get_logfn_name(mod), SIM->get_action_name(action));
|
|
}
|
|
}
|
|
// stage #2: save log actions of not yet existing modules (from bochsrc)
|
|
for (mod = 0; mod < loglev->get_size(); mod++) {
|
|
mparam = (bx_param_num_c*)loglev->get(mod);
|
|
action = mparam->get();
|
|
if ((action >= 0) && (action != def_action)) {
|
|
fprintf(fp, ", %s=%s", mparam->get_name(), SIM->get_action_name(action));
|
|
}
|
|
}
|
|
fprintf(fp, "\n");
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
int bx_write_debugger_options(FILE *fp)
|
|
{
|
|
#if BX_DEBUGGER
|
|
fprintf(fp, "debugger_log: %s\n", SIM->get_param_string(BXPN_DEBUGGER_LOG_FILENAME)->getptr());
|
|
fprintf(fp, "magic_break: enabled=1 0x%x\n", bx_dbg.magic_break);
|
|
// TODO: debug symbols
|
|
#endif
|
|
#if BX_GDBSTUB
|
|
bx_list_c *base = (bx_list_c*) SIM->get_param(BXPN_GDBSTUB);
|
|
bool enabled = SIM->get_param_bool("enabled", base)->get();
|
|
if (enabled) {
|
|
fprintf(fp, "gdbstub: enabled=%d, port=%d, text_base=%d, data_base=%d, bss_base=%d\n",
|
|
enabled, SIM->get_param_num("port", base)->get(), SIM->get_param_num("text_base", base)->get(),
|
|
SIM->get_param_num("data_base", base)->get(), SIM->get_param_num("bss_base", base)->get());
|
|
} else {
|
|
fprintf(fp, "# no gdb stub\n");
|
|
}
|
|
#endif
|
|
return 0;
|
|
}
|
|
|
|
// return values:
|
|
// 0: written ok
|
|
// -1: failed
|
|
// -2: already exists, and overwrite was off
|
|
int bx_write_configuration(const char *rc, int overwrite)
|
|
{
|
|
int i;
|
|
char pname[16], tmppath[80], tmpdev[80];
|
|
bx_param_string_c *sparam;
|
|
bx_list_c *base;
|
|
BX_INFO(("write current configuration to %s", rc));
|
|
// check if it exists. If so, only proceed if overwrite is set.
|
|
FILE *fp = fopen(rc, "r");
|
|
if (fp != NULL) {
|
|
fclose(fp);
|
|
if (!overwrite) return -2;
|
|
}
|
|
fp = fopen(rc, "w");
|
|
if (fp == NULL) return -1;
|
|
// finally it's open and we can start writing.
|
|
fprintf(fp, "# configuration file generated by Bochs\n");
|
|
bx_write_param_list(fp, (bx_list_c*) SIM->get_param(BXPN_PLUGIN_CTRL), NULL, 0);
|
|
fprintf(fp, "config_interface: %s\n", SIM->get_param_enum(BXPN_SEL_CONFIG_INTERFACE)->get_selected());
|
|
fprintf(fp, "display_library: %s", SIM->get_param_enum(BXPN_SEL_DISPLAY_LIBRARY)->get_selected());
|
|
sparam = SIM->get_param_string(BXPN_DISPLAYLIB_OPTIONS);
|
|
if (!sparam->isempty())
|
|
fprintf(fp, ", options=\"%s\"\n", sparam->getptr());
|
|
else
|
|
fprintf(fp, "\n");
|
|
|
|
bx_write_param_list(fp, (bx_list_c*) SIM->get_param(BXPN_MEMORY), "memory", 0);
|
|
bx_write_param_list(fp, (bx_list_c*) SIM->get_param(BXPN_ROMIMAGE), "romimage", 0);
|
|
bx_write_param_list(fp, (bx_list_c*) SIM->get_param(BXPN_VGA_ROMIMAGE), "vgaromimage", 0);
|
|
fprintf(fp, "boot: %s", SIM->get_param_enum(BXPN_BOOTDRIVE1)->get_selected());
|
|
for (i=1; i<3; i++) {
|
|
sprintf(tmppath, "boot_params.boot_drive%d", i+1);
|
|
if (SIM->get_param_enum(tmppath)->get() != BX_BOOT_NONE) {
|
|
fprintf(fp, ", %s", SIM->get_param_enum(tmppath)->get_selected());
|
|
}
|
|
}
|
|
fprintf(fp, "\n");
|
|
fprintf(fp, "floppy_bootsig_check: disabled=%d\n", SIM->get_param_bool(BXPN_FLOPPYSIGCHECK)->get());
|
|
// it would be nice to put this type of function as methods on
|
|
// the structs like bx_floppy_options::print or something.
|
|
bx_write_floppy_options(fp, 0);
|
|
bx_write_floppy_options(fp, 1);
|
|
for (Bit8u channel=0; channel<BX_MAX_ATA_CHANNEL; channel++) {
|
|
sprintf(tmppath, "ata.%d", channel);
|
|
base = (bx_list_c*) SIM->get_param(tmppath);
|
|
sprintf(tmppath, "ata%d", channel);
|
|
bx_write_param_list(fp, (bx_list_c*) SIM->get_param("resources", base), tmppath, 0);
|
|
sprintf(tmppath, "ata%d-master", channel);
|
|
bx_write_param_list(fp, (bx_list_c*) SIM->get_param("master", base), tmppath, 0);
|
|
sprintf(tmppath, "ata%d-slave", channel);
|
|
bx_write_param_list(fp, (bx_list_c*) SIM->get_param("slave", base), tmppath, 0);
|
|
}
|
|
for (i=0; i<BX_N_OPTROM_IMAGES; i++) {
|
|
sprintf(pname, "%s.%d", BXPN_OPTROM_BASE, i+1);
|
|
sprintf(tmppath, "optromimage%d", i+1);
|
|
bx_write_param_list(fp, (bx_list_c*) SIM->get_param(pname), tmppath, 0);
|
|
}
|
|
for (i=0; i<BX_N_OPTRAM_IMAGES; i++) {
|
|
sprintf(pname, "%s.%d", BXPN_OPTRAM_BASE, i+1);
|
|
sprintf(tmppath, "optramimage%d", i+1);
|
|
bx_write_param_list(fp, (bx_list_c*) SIM->get_param(pname), tmppath, 0);
|
|
}
|
|
// pci
|
|
fprintf(fp, "pci: enabled=%d",
|
|
SIM->get_param_bool(BXPN_PCI_ENABLED)->get());
|
|
if (SIM->get_param_bool(BXPN_PCI_ENABLED)->get()) {
|
|
fprintf(fp, ", chipset=%s", SIM->get_param_enum(BXPN_PCI_CHIPSET)->get_selected());
|
|
for (i=0; i<BX_N_PCI_SLOTS; i++) {
|
|
sprintf(tmpdev, "pci.slot.%d", i+1);
|
|
fprintf(fp, ", slot%d=%s", i+1, SIM->get_param_enum(tmpdev)->get_selected());
|
|
}
|
|
sparam = SIM->get_param_string(BXPN_PCI_ADV_OPTS);
|
|
if (strlen(sparam->getptr()) > 0) {
|
|
fprintf(fp, ", advopts=\"%s\"", sparam->getptr());
|
|
}
|
|
}
|
|
fprintf(fp, "\n");
|
|
fprintf(fp, "vga: extension=%s, update_freq=%u, realtime=%u, ddc=%s",
|
|
SIM->get_param_enum(BXPN_VGA_EXTENSION)->get_selected(),
|
|
SIM->get_param_num(BXPN_VGA_UPDATE_FREQUENCY)->get(),
|
|
SIM->get_param_bool(BXPN_VGA_REALTIME)->get(),
|
|
SIM->get_param_enum(BXPN_DDC_MODE)->get_selected());
|
|
if (SIM->get_param_enum(BXPN_DDC_MODE)->get() == BX_DDC_MODE_FILE) {
|
|
fprintf(fp, ":%s", SIM->get_param_string(BXPN_DDC_FILE)->getptr());
|
|
}
|
|
fprintf(fp, "\n");
|
|
#if BX_SUPPORT_SMP
|
|
fprintf(fp, "cpu: count=%u:%u:%u, ips=%u, quantum=%d, ",
|
|
SIM->get_param_num(BXPN_CPU_NPROCESSORS)->get(), SIM->get_param_num(BXPN_CPU_NCORES)->get(),
|
|
SIM->get_param_num(BXPN_CPU_NTHREADS)->get(), SIM->get_param_num(BXPN_IPS)->get(),
|
|
SIM->get_param_num(BXPN_SMP_QUANTUM)->get());
|
|
#else
|
|
fprintf(fp, "cpu: count=1, ips=%u, ", SIM->get_param_num(BXPN_IPS)->get());
|
|
#endif
|
|
fprintf(fp, "model=%s, reset_on_triple_fault=%d, cpuid_limit_winnt=%d",
|
|
SIM->get_param_enum(BXPN_CPU_MODEL)->get_selected(),
|
|
SIM->get_param_bool(BXPN_RESET_ON_TRIPLE_FAULT)->get(),
|
|
SIM->get_param_bool(BXPN_CPUID_LIMIT_WINNT)->get());
|
|
#if BX_CPU_LEVEL >= 5
|
|
fprintf(fp, ", ignore_bad_msrs=%d", SIM->get_param_bool(BXPN_IGNORE_BAD_MSRS)->get());
|
|
#endif
|
|
#if BX_SUPPORT_MONITOR_MWAIT
|
|
fprintf(fp, ", mwait_is_nop=%d", SIM->get_param_bool(BXPN_MWAIT_IS_NOP)->get());
|
|
#endif
|
|
#if BX_CONFIGURE_MSRS
|
|
sparam = SIM->get_param_string(BXPN_CONFIGURABLE_MSRS_PATH);
|
|
if (!sparam->isempty())
|
|
fprintf(fp, ", msrs=\"%s\"", sparam->getptr());
|
|
#endif
|
|
fprintf(fp, "\n");
|
|
|
|
#if BX_CPU_LEVEL >= 4
|
|
if (! SIM->get_param_enum(BXPN_CPU_MODEL)->get()) {
|
|
// dump only when using BX_GENERIC CPUDB profile
|
|
bx_write_param_list(fp, (bx_list_c*) SIM->get_param("cpuid"), NULL, 1);
|
|
}
|
|
#endif
|
|
|
|
fprintf(fp, "print_timestamps: enabled=%d\n", bx_dbg.print_timestamps);
|
|
bx_write_debugger_options(fp);
|
|
bx_write_param_list(fp, (bx_list_c*) SIM->get_param(BXPN_PORT_E9_HACK_ROOT), NULL, 0);
|
|
#if BX_SUPPORT_IODEBUG
|
|
fprintf(fp, "iodebug_all_rings: enabled=%d\n", SIM->get_param_bool(BXPN_IODEBUG_ALL_RINGS)->get());
|
|
#endif
|
|
fprintf(fp, "private_colormap: enabled=%d\n", SIM->get_param_bool(BXPN_PRIVATE_COLORMAP)->get());
|
|
#if BX_WITH_AMIGAOS
|
|
fprintf(fp, "fullscreen: enabled=%d\n", SIM->get_param_bool(BXPN_FULLSCREEN)->get());
|
|
fprintf(fp, "screenmode: name=\"%s\"\n", SIM->get_param_string(BXPN_SCREENMODE)->getptr());
|
|
#endif
|
|
bx_write_clock_cmos_options(fp);
|
|
bx_write_log_options(fp, (bx_list_c*) SIM->get_param("log"));
|
|
bx_write_param_list(fp, (bx_list_c*) SIM->get_param(BXPN_KEYBOARD), NULL, 0);
|
|
bx_write_param_list(fp, (bx_list_c*) SIM->get_param(BXPN_MOUSE), NULL, 0);
|
|
bx_write_param_list(fp, (bx_list_c*) SIM->get_param(BXPN_SOUNDLOW),"sound", 0);
|
|
SIM->save_addon_options(fp);
|
|
fclose(fp);
|
|
return 0;
|
|
}
|