2001-10-03 17:10:38 +04:00
|
|
|
/////////////////////////////////////////////////////////////////////////
|
2011-02-25 01:05:47 +03:00
|
|
|
// $Id$
|
2001-10-03 17:10:38 +04:00
|
|
|
/////////////////////////////////////////////////////////////////////////
|
|
|
|
//
|
2009-12-04 22:50:29 +03:00
|
|
|
// Copyright (C) 2004-2009 The Bochs Project
|
2001-04-10 05:04:59 +04:00
|
|
|
//
|
|
|
|
// This library is free software; you can redistribute it and/or
|
|
|
|
// modify it under the terms of the GNU Lesser General Public
|
|
|
|
// License as published by the Free Software Foundation; either
|
|
|
|
// version 2 of the License, or (at your option) any later version.
|
|
|
|
//
|
|
|
|
// This library is distributed in the hope that it will be useful,
|
|
|
|
// but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
|
|
// Lesser General Public License for more details.
|
|
|
|
//
|
|
|
|
// You should have received a copy of the GNU Lesser General Public
|
|
|
|
// License along with this library; if not, write to the Free Software
|
2009-02-08 12:05:52 +03:00
|
|
|
// Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
|
2008-02-16 01:05:43 +03:00
|
|
|
/////////////////////////////////////////////////////////////////////////
|
2001-04-10 05:04:59 +04:00
|
|
|
|
2003-11-09 03:14:43 +03:00
|
|
|
// Peter Grehan (grehan@iprg.nokia.com) coded the original version of this
|
|
|
|
// serial emulation. He implemented a single 8250, and allow terminal
|
|
|
|
// input/output to stdout on FreeBSD.
|
2004-01-18 14:58:07 +03:00
|
|
|
// The current version emulates up to 4 UART 16550A with FIFO. Terminal
|
2003-11-09 03:14:43 +03:00
|
|
|
// input/output now works on some more platforms.
|
2001-04-10 05:04:59 +04:00
|
|
|
|
2002-10-25 01:07:56 +04:00
|
|
|
// Define BX_PLUGGABLE in files that can be compiled into plugins. For
|
2008-01-27 01:24:03 +03:00
|
|
|
// platforms that require a special tag on exported symbols, BX_PLUGGABLE
|
2002-10-25 01:07:56 +04:00
|
|
|
// is used to know when we are exporting symbols and when we are importing.
|
|
|
|
#define BX_PLUGGABLE
|
|
|
|
|
2005-07-10 20:51:09 +04:00
|
|
|
#ifndef WIN32
|
2011-04-19 16:48:06 +04:00
|
|
|
#include <sys/types.h>
|
2005-07-10 20:51:09 +04:00
|
|
|
#include <sys/socket.h>
|
2005-07-11 20:24:47 +04:00
|
|
|
#include <netinet/in.h>
|
2005-07-10 20:51:09 +04:00
|
|
|
#include <netdb.h>
|
2008-05-22 12:13:22 +04:00
|
|
|
#define closesocket(s) close(s)
|
2010-11-23 23:26:37 +03:00
|
|
|
typedef int SOCKET;
|
2005-07-10 20:51:09 +04:00
|
|
|
#endif
|
2002-10-25 01:07:56 +04:00
|
|
|
|
2010-11-23 23:26:37 +03:00
|
|
|
#include "iodev.h"
|
|
|
|
#include "serial.h"
|
|
|
|
|
2011-01-15 01:15:37 +03:00
|
|
|
#if defined(WIN32) && !defined(FILE_FLAG_FIRST_PIPE_INSTANCE)
|
|
|
|
#define FILE_FLAG_FIRST_PIPE_INSTANCE 0
|
|
|
|
#endif
|
|
|
|
|
2004-09-05 14:30:19 +04:00
|
|
|
#if USE_RAW_SERIAL
|
|
|
|
#include "serial_raw.h"
|
2011-01-15 01:15:37 +03:00
|
|
|
#endif
|
2004-09-05 14:30:19 +04:00
|
|
|
|
2002-10-25 01:07:56 +04:00
|
|
|
#define LOG_THIS theSerialDevice->
|
2001-04-10 05:04:59 +04:00
|
|
|
|
2002-10-25 01:07:56 +04:00
|
|
|
bx_serial_c *theSerialDevice = NULL;
|
|
|
|
|
2012-04-16 23:17:10 +04:00
|
|
|
// builtin configuration handling functions
|
|
|
|
|
|
|
|
void serial_init_options(void)
|
|
|
|
{
|
|
|
|
static const char *serial_mode_list[] = {
|
|
|
|
"null",
|
|
|
|
"file",
|
|
|
|
"pipe",
|
|
|
|
"pipe-client",
|
|
|
|
"pipe-server",
|
|
|
|
"term",
|
|
|
|
"raw",
|
|
|
|
"mouse",
|
|
|
|
"socket",
|
|
|
|
"socket-client",
|
|
|
|
"socket-server",
|
|
|
|
NULL
|
|
|
|
};
|
|
|
|
|
2012-04-21 22:13:10 +04:00
|
|
|
char name[4], label[80], descr[120];
|
2012-04-16 23:17:10 +04:00
|
|
|
|
|
|
|
bx_list_c *serial = (bx_list_c*)SIM->get_param("ports.serial");
|
|
|
|
for (int i=0; i<BX_N_SERIAL_PORTS; i++) {
|
|
|
|
sprintf(name, "%d", i+1);
|
|
|
|
sprintf(label, "Serial Port %d", i+1);
|
|
|
|
bx_list_c *menu = new bx_list_c(serial, name, label);
|
|
|
|
menu->set_options(menu->SERIES_ASK);
|
|
|
|
sprintf(label, "Enable serial port #%d (COM%d)", i+1, i+1);
|
|
|
|
sprintf(descr, "Controls whether COM%d is installed or not", i+1);
|
|
|
|
bx_param_bool_c *enabled = new bx_param_bool_c(menu, "enabled", label, descr,
|
|
|
|
(i==0)?1 : 0); // only enable the first by default
|
|
|
|
sprintf(label, "I/O mode of the serial device for COM%d", i+1);
|
2012-04-21 22:13:10 +04:00
|
|
|
bx_param_enum_c *mode = new bx_param_enum_c(menu, "mode", label,
|
|
|
|
"The mode can be one these: 'null', 'file', 'pipe', 'term', 'raw', 'mouse', 'socket'",
|
2012-04-16 23:17:10 +04:00
|
|
|
serial_mode_list, 0, 0);
|
|
|
|
mode->set_ask_format("Choose I/O mode of the serial device [%s] ");
|
|
|
|
sprintf(label, "Pathname of the serial device for COM%d", i+1);
|
2012-04-21 22:13:10 +04:00
|
|
|
bx_param_filename_c *path = new bx_param_filename_c(menu, "dev", label,
|
|
|
|
"The path can be a real serial device or a pty (X/Unix only)",
|
2012-04-16 23:17:10 +04:00
|
|
|
"", BX_PATHNAME_LEN);
|
|
|
|
bx_list_c *deplist = new bx_list_c(NULL);
|
|
|
|
deplist->add(mode);
|
|
|
|
deplist->add(path);
|
|
|
|
enabled->set_dependent_list(deplist);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
Bit32s serial_options_parser(const char *context, int num_params, char *params[])
|
|
|
|
{
|
|
|
|
if ((!strncmp(params[0], "com", 3)) && (strlen(params[0]) == 4)) {
|
|
|
|
char tmpname[80];
|
|
|
|
int idx = params[0][3];
|
|
|
|
if ((idx < '1') || (idx > '9')) {
|
|
|
|
BX_PANIC(("%s: comX directive malformed.", context));
|
|
|
|
}
|
|
|
|
idx -= '0';
|
|
|
|
if (idx > BX_N_SERIAL_PORTS) {
|
|
|
|
BX_PANIC(("%s: comX port number out of range.", context));
|
|
|
|
}
|
|
|
|
sprintf(tmpname, "ports.serial.%d", idx);
|
|
|
|
bx_list_c *base = (bx_list_c*) SIM->get_param(tmpname);
|
|
|
|
for (int i=1; i<num_params; i++) {
|
|
|
|
if (!strncmp(params[i], "enabled=", 8)) {
|
|
|
|
SIM->get_param_bool("enabled", base)->set(atol(¶ms[i][8]));
|
|
|
|
} else if (!strncmp(params[i], "mode=", 5)) {
|
|
|
|
if (!SIM->get_param_enum("mode", base)->set_by_name(¶ms[i][5]))
|
|
|
|
BX_PANIC(("%s: com%d serial port mode '%s' not available", context, idx, ¶ms[i][5]));
|
|
|
|
SIM->get_param_bool("enabled", base)->set(1);
|
|
|
|
} else if (!strncmp(params[i], "dev=", 4)) {
|
|
|
|
SIM->get_param_string("dev", base)->set(¶ms[i][4]);
|
|
|
|
SIM->get_param_bool("enabled", base)->set(1);
|
|
|
|
} else {
|
|
|
|
BX_ERROR(("%s: unknown parameter for com%d ignored.", context, idx));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
BX_PANIC(("%s: unknown directive '%s'", context, params[0]));
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
Bit32s serial_options_save(FILE *fp)
|
|
|
|
{
|
|
|
|
char pname[20];
|
|
|
|
|
|
|
|
for (int i=0; i<BX_N_SERIAL_PORTS; i++) {
|
|
|
|
sprintf(pname, "ports.serial.%d", i+1);
|
|
|
|
bx_list_c *base = (bx_list_c*) SIM->get_param(pname);
|
|
|
|
fprintf(fp, "com%d: enabled=%d", i+1, SIM->get_param_bool("enabled", base)->get());
|
|
|
|
if (SIM->get_param_bool("enabled", base)->get()) {
|
|
|
|
fprintf(fp, ", mode=%s", SIM->get_param_enum("mode", base)->get_selected());
|
|
|
|
fprintf(fp, ", dev=\"%s\"", SIM->get_param_string("dev", base)->getptr());
|
|
|
|
}
|
|
|
|
fprintf(fp, "\n");
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
// device plugin entry points
|
|
|
|
|
2006-09-10 21:18:44 +04:00
|
|
|
int libserial_LTX_plugin_init(plugin_t *plugin, plugintype_t type, int argc, char *argv[])
|
2002-10-25 01:07:56 +04:00
|
|
|
{
|
2006-09-10 21:18:44 +04:00
|
|
|
theSerialDevice = new bx_serial_c();
|
2002-10-25 01:07:56 +04:00
|
|
|
BX_REGISTER_DEVICE_DEVMODEL(plugin, type, theSerialDevice, BX_PLUGIN_SERIAL);
|
2012-04-16 23:17:10 +04:00
|
|
|
// add new configuration parameters for the config interface
|
|
|
|
serial_init_options();
|
|
|
|
// register add-on options for bochsrc and command line
|
|
|
|
SIM->register_addon_option("com1", serial_options_parser, serial_options_save);
|
|
|
|
SIM->register_addon_option("com2", serial_options_parser, NULL);
|
|
|
|
SIM->register_addon_option("com3", serial_options_parser, NULL);
|
|
|
|
SIM->register_addon_option("com4", serial_options_parser, NULL);
|
2002-10-25 01:07:56 +04:00
|
|
|
return(0); // Success
|
|
|
|
}
|
|
|
|
|
2006-09-10 21:18:44 +04:00
|
|
|
void libserial_LTX_plugin_fini(void)
|
2002-10-25 01:07:56 +04:00
|
|
|
{
|
2012-04-16 23:17:10 +04:00
|
|
|
char pnum[4];
|
|
|
|
|
|
|
|
SIM->unregister_addon_option("com1");
|
|
|
|
SIM->unregister_addon_option("com2");
|
|
|
|
SIM->unregister_addon_option("com3");
|
|
|
|
SIM->unregister_addon_option("com4");
|
2006-09-10 21:18:44 +04:00
|
|
|
delete theSerialDevice;
|
2012-04-16 23:17:10 +04:00
|
|
|
bx_list_c *menu = (bx_list_c*)SIM->get_param("ports.serial");
|
|
|
|
for (int i=0; i<BX_N_SERIAL_PORTS; i++) {
|
|
|
|
sprintf(pnum, "%d", i+1);
|
|
|
|
menu->remove(pnum);
|
|
|
|
}
|
2002-10-25 01:07:56 +04:00
|
|
|
}
|
|
|
|
|
2012-04-16 23:17:10 +04:00
|
|
|
// the device object
|
|
|
|
|
2001-04-10 05:04:59 +04:00
|
|
|
bx_serial_c::bx_serial_c(void)
|
|
|
|
{
|
2011-12-29 23:51:54 +04:00
|
|
|
put("serial", "SER");
|
2002-10-06 23:04:47 +04:00
|
|
|
for (int i=0; i<BX_SERIAL_MAXDEV; i++) {
|
2012-02-05 14:08:56 +04:00
|
|
|
memset(&s[i], 0, sizeof(bx_serial_t));
|
2006-09-12 17:05:07 +04:00
|
|
|
s[i].io_mode = BX_SER_MODE_NULL;
|
2004-01-18 03:18:44 +03:00
|
|
|
s[i].tty_id = -1;
|
2002-10-06 23:04:47 +04:00
|
|
|
s[i].tx_timer_index = BX_NULL_TIMER_HANDLE;
|
|
|
|
s[i].rx_timer_index = BX_NULL_TIMER_HANDLE;
|
2003-11-09 03:14:43 +03:00
|
|
|
s[i].fifo_timer_index = BX_NULL_TIMER_HANDLE;
|
2002-10-06 23:04:47 +04:00
|
|
|
}
|
2002-03-03 09:03:29 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
bx_serial_c::~bx_serial_c(void)
|
|
|
|
{
|
2006-03-01 20:14:36 +03:00
|
|
|
char pname[20];
|
|
|
|
bx_list_c *base;
|
|
|
|
|
2004-01-18 03:18:44 +03:00
|
|
|
for (int i=0; i<BX_SERIAL_MAXDEV; i++) {
|
2006-03-01 20:14:36 +03:00
|
|
|
sprintf(pname, "ports.serial.%d", i+1);
|
|
|
|
base = (bx_list_c*) SIM->get_param(pname);
|
|
|
|
if (SIM->get_param_bool("enabled", base)->get()) {
|
2004-07-28 23:36:42 +04:00
|
|
|
switch (BX_SER_THIS s[i].io_mode) {
|
|
|
|
case BX_SER_MODE_FILE:
|
|
|
|
if (BX_SER_THIS s[i].output != NULL)
|
|
|
|
fclose(BX_SER_THIS s[i].output);
|
|
|
|
break;
|
|
|
|
case BX_SER_MODE_TERM:
|
2005-07-10 20:51:09 +04:00
|
|
|
#if defined(SERIAL_ENABLE) && !defined(WIN32)
|
2004-07-28 23:36:42 +04:00
|
|
|
if (s[i].tty_id >= 0) {
|
|
|
|
tcsetattr(s[i].tty_id, TCSAFLUSH, &s[i].term_orig);
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
break;
|
|
|
|
case BX_SER_MODE_RAW:
|
2004-02-29 00:28:28 +03:00
|
|
|
#if USE_RAW_SERIAL
|
2004-07-28 23:36:42 +04:00
|
|
|
delete [] BX_SER_THIS s[i].raw;
|
2002-03-03 09:03:29 +03:00
|
|
|
#endif
|
2004-07-28 23:36:42 +04:00
|
|
|
break;
|
2005-07-10 20:51:09 +04:00
|
|
|
case BX_SER_MODE_SOCKET:
|
2008-05-22 12:13:22 +04:00
|
|
|
if (BX_SER_THIS s[i].socket_id >= 0) closesocket(BX_SER_THIS s[i].socket_id);
|
|
|
|
break;
|
|
|
|
case BX_SER_MODE_PIPE:
|
|
|
|
#ifdef WIN32
|
|
|
|
if (BX_SER_THIS s[i].pipe)
|
|
|
|
CloseHandle(BX_SER_THIS s[i].pipe);
|
|
|
|
#endif
|
2005-07-10 20:51:09 +04:00
|
|
|
break;
|
2004-07-28 23:36:42 +04:00
|
|
|
}
|
2004-02-29 01:06:36 +03:00
|
|
|
}
|
2004-02-29 00:28:28 +03:00
|
|
|
}
|
2006-09-10 21:18:44 +04:00
|
|
|
BX_DEBUG(("Exit"));
|
2002-03-03 09:03:29 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void
|
2002-10-25 01:07:56 +04:00
|
|
|
bx_serial_c::init(void)
|
2002-03-03 09:03:29 +03:00
|
|
|
{
|
2003-10-30 00:00:04 +03:00
|
|
|
Bit16u ports[BX_SERIAL_MAXDEV] = {0x03f8, 0x02f8, 0x03e8, 0x02e8};
|
2006-03-01 20:14:36 +03:00
|
|
|
char name[16], pname[20];
|
|
|
|
bx_list_c *base;
|
2012-04-16 23:17:10 +04:00
|
|
|
unsigned i, count = 0;
|
2003-10-30 00:00:04 +03:00
|
|
|
|
2004-12-05 23:23:39 +03:00
|
|
|
BX_SER_THIS detect_mouse = 0;
|
2004-12-03 00:34:26 +03:00
|
|
|
BX_SER_THIS mouse_port = -1;
|
2006-12-31 14:56:14 +03:00
|
|
|
BX_SER_THIS mouse_type = BX_MOUSE_TYPE_NONE;
|
2004-12-03 00:34:26 +03:00
|
|
|
BX_SER_THIS mouse_internal_buffer.num_elements = 0;
|
2005-01-19 21:21:40 +03:00
|
|
|
for (i=0; i<BX_MOUSE_BUFF_SIZE; i++)
|
2004-12-03 00:34:26 +03:00
|
|
|
BX_SER_THIS mouse_internal_buffer.buffer[i] = 0;
|
|
|
|
BX_SER_THIS mouse_internal_buffer.head = 0;
|
|
|
|
BX_SER_THIS mouse_delayed_dx = 0;
|
|
|
|
BX_SER_THIS mouse_delayed_dy = 0;
|
2001-04-10 05:04:59 +04:00
|
|
|
/*
|
|
|
|
* Put the UART registers into their RESET state
|
|
|
|
*/
|
2005-01-19 21:21:40 +03:00
|
|
|
for (i=0; i<BX_N_SERIAL_PORTS; i++) {
|
2006-03-01 20:14:36 +03:00
|
|
|
sprintf(pname, "ports.serial.%d", i+1);
|
|
|
|
base = (bx_list_c*) SIM->get_param(pname);
|
|
|
|
if (SIM->get_param_bool("enabled", base)->get()) {
|
2003-10-30 00:00:04 +03:00
|
|
|
sprintf(name, "Serial Port %d", i + 1);
|
|
|
|
/* serial interrupt */
|
|
|
|
BX_SER_THIS s[i].IRQ = 4 - (i & 1);
|
|
|
|
if (i < 2) {
|
|
|
|
DEV_register_irq(BX_SER_THIS s[i].IRQ, name);
|
|
|
|
}
|
|
|
|
/* internal state */
|
|
|
|
BX_SER_THIS s[i].ls_ipending = 0;
|
|
|
|
BX_SER_THIS s[i].ms_ipending = 0;
|
|
|
|
BX_SER_THIS s[i].rx_ipending = 0;
|
2003-11-09 03:14:43 +03:00
|
|
|
BX_SER_THIS s[i].fifo_ipending = 0;
|
2003-10-30 00:00:04 +03:00
|
|
|
BX_SER_THIS s[i].ls_interrupt = 0;
|
|
|
|
BX_SER_THIS s[i].ms_interrupt = 0;
|
|
|
|
BX_SER_THIS s[i].rx_interrupt = 0;
|
|
|
|
BX_SER_THIS s[i].tx_interrupt = 0;
|
2003-11-09 03:14:43 +03:00
|
|
|
BX_SER_THIS s[i].fifo_interrupt = 0;
|
2003-10-30 00:00:04 +03:00
|
|
|
|
|
|
|
if (BX_SER_THIS s[i].tx_timer_index == BX_NULL_TIMER_HANDLE) {
|
|
|
|
BX_SER_THIS s[i].tx_timer_index =
|
|
|
|
bx_pc_system.register_timer(this, tx_timer_handler, 0,
|
|
|
|
0,0, "serial.tx"); // one-shot, inactive
|
|
|
|
}
|
2002-10-06 23:04:47 +04:00
|
|
|
|
2003-10-30 00:00:04 +03:00
|
|
|
if (BX_SER_THIS s[i].rx_timer_index == BX_NULL_TIMER_HANDLE) {
|
|
|
|
BX_SER_THIS s[i].rx_timer_index =
|
|
|
|
bx_pc_system.register_timer(this, rx_timer_handler, 0,
|
|
|
|
0,0, "serial.rx"); // one-shot, inactive
|
|
|
|
}
|
2003-11-09 03:14:43 +03:00
|
|
|
if (BX_SER_THIS s[i].fifo_timer_index == BX_NULL_TIMER_HANDLE) {
|
|
|
|
BX_SER_THIS s[i].fifo_timer_index =
|
|
|
|
bx_pc_system.register_timer(this, fifo_timer_handler, 0,
|
|
|
|
0,0, "serial.fifo"); // one-shot, inactive
|
|
|
|
}
|
2003-10-30 00:00:04 +03:00
|
|
|
BX_SER_THIS s[i].rx_pollstate = BX_SER_RXIDLE;
|
|
|
|
|
|
|
|
/* int enable: b0000 0000 */
|
|
|
|
BX_SER_THIS s[i].int_enable.rxdata_enable = 0;
|
|
|
|
BX_SER_THIS s[i].int_enable.txhold_enable = 0;
|
|
|
|
BX_SER_THIS s[i].int_enable.rxlstat_enable = 0;
|
|
|
|
BX_SER_THIS s[i].int_enable.modstat_enable = 0;
|
|
|
|
|
|
|
|
/* int ID: b0000 0001 */
|
|
|
|
BX_SER_THIS s[i].int_ident.ipending = 1;
|
|
|
|
BX_SER_THIS s[i].int_ident.int_ID = 0;
|
|
|
|
|
|
|
|
/* FIFO control: b0000 0000 */
|
|
|
|
BX_SER_THIS s[i].fifo_cntl.enable = 0;
|
|
|
|
BX_SER_THIS s[i].fifo_cntl.rxtrigger = 0;
|
2003-11-09 03:14:43 +03:00
|
|
|
BX_SER_THIS s[i].rx_fifo_end = 0;
|
|
|
|
BX_SER_THIS s[i].tx_fifo_end = 0;
|
2003-10-30 00:00:04 +03:00
|
|
|
|
|
|
|
/* Line Control reg: b0000 0000 */
|
|
|
|
BX_SER_THIS s[i].line_cntl.wordlen_sel = 0;
|
|
|
|
BX_SER_THIS s[i].line_cntl.stopbits = 0;
|
|
|
|
BX_SER_THIS s[i].line_cntl.parity_enable = 0;
|
|
|
|
BX_SER_THIS s[i].line_cntl.evenparity_sel = 0;
|
|
|
|
BX_SER_THIS s[i].line_cntl.stick_parity = 0;
|
|
|
|
BX_SER_THIS s[i].line_cntl.break_cntl = 0;
|
|
|
|
BX_SER_THIS s[i].line_cntl.dlab = 0;
|
|
|
|
|
|
|
|
/* Modem Control reg: b0000 0000 */
|
|
|
|
BX_SER_THIS s[i].modem_cntl.dtr = 0;
|
|
|
|
BX_SER_THIS s[i].modem_cntl.rts = 0;
|
|
|
|
BX_SER_THIS s[i].modem_cntl.out1 = 0;
|
|
|
|
BX_SER_THIS s[i].modem_cntl.out2 = 0;
|
|
|
|
BX_SER_THIS s[i].modem_cntl.local_loopback = 0;
|
|
|
|
|
|
|
|
/* Line Status register: b0110 0000 */
|
|
|
|
BX_SER_THIS s[i].line_status.rxdata_ready = 0;
|
|
|
|
BX_SER_THIS s[i].line_status.overrun_error = 0;
|
|
|
|
BX_SER_THIS s[i].line_status.parity_error = 0;
|
|
|
|
BX_SER_THIS s[i].line_status.framing_error = 0;
|
|
|
|
BX_SER_THIS s[i].line_status.break_int = 0;
|
|
|
|
BX_SER_THIS s[i].line_status.thr_empty = 1;
|
|
|
|
BX_SER_THIS s[i].line_status.tsr_empty = 1;
|
|
|
|
BX_SER_THIS s[i].line_status.fifo_error = 0;
|
|
|
|
|
|
|
|
/* Modem Status register: bXXXX 0000 */
|
|
|
|
BX_SER_THIS s[i].modem_status.delta_cts = 0;
|
|
|
|
BX_SER_THIS s[i].modem_status.delta_dsr = 0;
|
|
|
|
BX_SER_THIS s[i].modem_status.ri_trailedge = 0;
|
|
|
|
BX_SER_THIS s[i].modem_status.delta_dcd = 0;
|
|
|
|
BX_SER_THIS s[i].modem_status.cts = 0;
|
|
|
|
BX_SER_THIS s[i].modem_status.dsr = 0;
|
|
|
|
BX_SER_THIS s[i].modem_status.ri = 0;
|
|
|
|
BX_SER_THIS s[i].modem_status.dcd = 0;
|
|
|
|
|
|
|
|
BX_SER_THIS s[i].scratch = 0; /* scratch register */
|
2003-10-31 20:23:56 +03:00
|
|
|
BX_SER_THIS s[i].divisor_lsb = 1; /* divisor-lsb register */
|
2003-10-30 00:00:04 +03:00
|
|
|
BX_SER_THIS s[i].divisor_msb = 0; /* divisor-msb register */
|
|
|
|
|
2003-10-31 20:23:56 +03:00
|
|
|
BX_SER_THIS s[i].baudrate = 115200;
|
|
|
|
|
2003-10-30 00:00:04 +03:00
|
|
|
for (unsigned addr=ports[i]; addr<(unsigned)(ports[i]+8); addr++) {
|
2005-01-02 13:42:15 +03:00
|
|
|
BX_DEBUG(("com%d initialize register for read/write: 0x%04x",i+1, addr));
|
2003-10-30 00:00:04 +03:00
|
|
|
DEV_register_ioread_handler(this, read_handler, addr, name, 1);
|
|
|
|
DEV_register_iowrite_handler(this, write_handler, addr, name, 1);
|
|
|
|
}
|
2004-07-28 23:36:42 +04:00
|
|
|
|
|
|
|
BX_SER_THIS s[i].io_mode = BX_SER_MODE_NULL;
|
2007-10-25 03:17:42 +04:00
|
|
|
const char *mode = SIM->get_param_enum("mode", base)->get_selected();
|
|
|
|
const char *dev = SIM->get_param_string("dev", base)->getptr();
|
2004-07-28 23:36:42 +04:00
|
|
|
if (!strcmp(mode, "file")) {
|
2006-03-01 20:14:36 +03:00
|
|
|
if (strlen(dev) > 0) {
|
|
|
|
BX_SER_THIS s[i].output = fopen(dev, "wb");
|
2004-07-28 23:36:42 +04:00
|
|
|
if (BX_SER_THIS s[i].output)
|
|
|
|
BX_SER_THIS s[i].io_mode = BX_SER_MODE_FILE;
|
|
|
|
}
|
|
|
|
} else if (!strcmp(mode, "term")) {
|
2005-07-10 20:51:09 +04:00
|
|
|
#if defined(SERIAL_ENABLE) && !defined(WIN32)
|
2006-03-01 20:14:36 +03:00
|
|
|
if (strlen(dev) > 0) {
|
|
|
|
BX_SER_THIS s[i].tty_id = open(dev, O_RDWR|O_NONBLOCK,600);
|
2004-07-28 23:36:42 +04:00
|
|
|
if (BX_SER_THIS s[i].tty_id < 0) {
|
2006-03-01 20:14:36 +03:00
|
|
|
BX_PANIC(("open of com%d (%s) failed", i+1, dev));
|
2004-07-28 23:36:42 +04:00
|
|
|
} else {
|
|
|
|
BX_SER_THIS s[i].io_mode = BX_SER_MODE_TERM;
|
|
|
|
BX_DEBUG(("com%d tty_id: %d", i+1, BX_SER_THIS s[i].tty_id));
|
|
|
|
tcgetattr(BX_SER_THIS s[i].tty_id, &BX_SER_THIS s[i].term_orig);
|
2006-10-08 14:59:21 +04:00
|
|
|
memcpy(&BX_SER_THIS s[i].term_orig, &BX_SER_THIS s[i].term_new, sizeof(struct termios));
|
|
|
|
BX_SER_THIS s[i].term_new.c_iflag &= ~(IGNBRK|BRKINT|PARMRK|ISTRIP|INLCR|IGNCR|ICRNL|IXON);
|
|
|
|
BX_SER_THIS s[i].term_new.c_lflag &= ~(ECHO|ECHONL|ICANON|ISIG|IEXTEN);
|
|
|
|
BX_SER_THIS s[i].term_new.c_cflag &= ~(CSIZE|PARENB);
|
|
|
|
BX_SER_THIS s[i].term_new.c_cflag |= CS8;
|
2004-07-28 23:36:42 +04:00
|
|
|
BX_SER_THIS s[i].term_new.c_oflag |= OPOST | ONLCR; // Enable NL to CR-NL translation
|
2004-01-18 03:18:44 +03:00
|
|
|
#ifndef TRUE_CTLC
|
2004-07-28 23:36:42 +04:00
|
|
|
// ctl-C will exit Bochs, or trap to the debugger
|
|
|
|
BX_SER_THIS s[i].term_new.c_iflag &= ~IGNBRK;
|
|
|
|
BX_SER_THIS s[i].term_new.c_iflag |= BRKINT;
|
|
|
|
BX_SER_THIS s[i].term_new.c_lflag |= ISIG;
|
2004-01-18 03:18:44 +03:00
|
|
|
#else
|
2004-07-28 23:36:42 +04:00
|
|
|
// ctl-C will be delivered to the serial port
|
|
|
|
BX_SER_THIS s[i].term_new.c_iflag |= IGNBRK;
|
|
|
|
BX_SER_THIS s[i].term_new.c_iflag &= ~BRKINT;
|
2004-01-18 03:18:44 +03:00
|
|
|
#endif /* !def TRUE_CTLC */
|
2004-07-28 23:36:42 +04:00
|
|
|
BX_SER_THIS s[i].term_new.c_iflag = 0;
|
|
|
|
BX_SER_THIS s[i].term_new.c_oflag = 0;
|
|
|
|
BX_SER_THIS s[i].term_new.c_cflag = CS8|CREAD|CLOCAL;
|
|
|
|
BX_SER_THIS s[i].term_new.c_lflag = 0;
|
|
|
|
BX_SER_THIS s[i].term_new.c_cc[VMIN] = 1;
|
|
|
|
BX_SER_THIS s[i].term_new.c_cc[VTIME] = 0;
|
|
|
|
//BX_SER_THIS s[i].term_new.c_iflag |= IXOFF;
|
|
|
|
tcsetattr(BX_SER_THIS s[i].tty_id, TCSAFLUSH, &BX_SER_THIS s[i].term_new);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#else
|
|
|
|
BX_PANIC(("serial terminal support not available"));
|
2004-01-18 03:18:44 +03:00
|
|
|
#endif /* def SERIAL_ENABLE */
|
2004-07-28 23:36:42 +04:00
|
|
|
} else if (!strcmp(mode, "raw")) {
|
|
|
|
#if USE_RAW_SERIAL
|
2006-03-01 20:14:36 +03:00
|
|
|
BX_SER_THIS s[i].raw = new serial_raw(dev);
|
2004-07-28 23:36:42 +04:00
|
|
|
BX_SER_THIS s[i].io_mode = BX_SER_MODE_RAW;
|
|
|
|
#else
|
|
|
|
BX_PANIC(("raw serial support not present"));
|
|
|
|
#endif
|
2004-11-27 13:09:41 +03:00
|
|
|
} else if (!strcmp(mode, "mouse")) {
|
|
|
|
BX_SER_THIS s[i].io_mode = BX_SER_MODE_MOUSE;
|
2004-12-03 00:34:26 +03:00
|
|
|
BX_SER_THIS mouse_port = i;
|
2006-12-31 14:56:14 +03:00
|
|
|
BX_SER_THIS mouse_type = SIM->get_param_enum(BXPN_MOUSE_TYPE)->get();
|
2008-05-22 12:13:22 +04:00
|
|
|
} else if (!strncmp(mode, "socket", 6)) {
|
2005-07-10 20:51:09 +04:00
|
|
|
BX_SER_THIS s[i].io_mode = BX_SER_MODE_SOCKET;
|
|
|
|
struct sockaddr_in sin;
|
|
|
|
struct hostent *hp;
|
|
|
|
char host[BX_PATHNAME_LEN];
|
|
|
|
int port;
|
2010-11-23 17:59:36 +03:00
|
|
|
SOCKET socket;
|
2008-05-22 12:13:22 +04:00
|
|
|
bx_bool server = !strcmp(mode, "socket-server");
|
2005-07-10 20:51:09 +04:00
|
|
|
|
|
|
|
#if defined(WIN32)
|
2008-05-22 12:13:22 +04:00
|
|
|
static bx_bool winsock_init = false;
|
2005-07-10 20:51:09 +04:00
|
|
|
if (!winsock_init) {
|
|
|
|
WORD wVersionRequested;
|
|
|
|
WSADATA wsaData;
|
|
|
|
int err;
|
2006-08-18 20:57:39 +04:00
|
|
|
wVersionRequested = MAKEWORD(2, 0);
|
|
|
|
err = WSAStartup(wVersionRequested, &wsaData);
|
2005-07-10 20:51:09 +04:00
|
|
|
if (err != 0)
|
2006-08-18 20:57:39 +04:00
|
|
|
BX_PANIC(("WSAStartup failed"));
|
2005-07-10 20:51:09 +04:00
|
|
|
winsock_init = true;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2006-08-18 20:57:39 +04:00
|
|
|
strcpy(host, dev);
|
|
|
|
char *substr = strtok(host, ":");
|
2005-07-10 20:51:09 +04:00
|
|
|
substr = strtok(NULL, ":");
|
|
|
|
if (!substr) {
|
2006-03-01 20:14:36 +03:00
|
|
|
BX_PANIC(("com%d: inet address is wrong (%s)", i+1, dev));
|
2005-07-10 20:51:09 +04:00
|
|
|
}
|
2006-08-18 20:57:39 +04:00
|
|
|
port = atoi(substr);
|
2005-07-10 20:51:09 +04:00
|
|
|
|
2006-08-18 20:57:39 +04:00
|
|
|
hp = gethostbyname(host);
|
2005-07-10 20:51:09 +04:00
|
|
|
if (!hp) {
|
|
|
|
BX_PANIC(("com%d: gethostbyname failed (%s)", i+1, host));
|
|
|
|
}
|
|
|
|
|
|
|
|
memset ((char*) &sin, 0, sizeof (sin));
|
2005-07-11 20:24:47 +04:00
|
|
|
#if BX_HAVE_SOCKADDR_IN_SIN_LEN
|
|
|
|
sin.sin_len = sizeof sin;
|
|
|
|
#endif
|
2005-07-10 20:51:09 +04:00
|
|
|
memcpy ((char*) &(sin.sin_addr), hp->h_addr, hp->h_length);
|
|
|
|
sin.sin_family = hp->h_addrtype;
|
|
|
|
sin.sin_port = htons (port);
|
|
|
|
|
|
|
|
socket = ::socket (AF_INET, SOCK_STREAM, 0);
|
2008-05-22 12:13:22 +04:00
|
|
|
if (socket < 0)
|
2005-07-10 20:51:09 +04:00
|
|
|
BX_PANIC(("com%d: socket() failed",i+1));
|
|
|
|
|
2008-05-22 12:13:22 +04:00
|
|
|
// server mode
|
|
|
|
if (server) {
|
|
|
|
if (::bind (socket, (sockaddr *) &sin, sizeof (sin)) < 0 ||
|
|
|
|
::listen (socket, SOMAXCONN) < 0) {
|
|
|
|
closesocket(socket);
|
2010-11-23 17:59:36 +03:00
|
|
|
socket = (SOCKET) -1;
|
2008-05-22 12:13:22 +04:00
|
|
|
BX_PANIC(("com%d: bind() or listen() failed (host:%s, port:%d)",i+1, host, port));
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
BX_INFO(("com%d: waiting for client to connect (host:%s, port:%d)",i+1, host, port));
|
2010-11-23 17:59:36 +03:00
|
|
|
SOCKET client;
|
2008-05-22 12:13:22 +04:00
|
|
|
if ((client = ::accept (socket, NULL, 0)) < 0)
|
|
|
|
BX_PANIC(("com%d: accept() failed (host:%s, port:%d)",i+1, host, port));
|
|
|
|
closesocket(socket);
|
|
|
|
socket = client;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
// client mode
|
|
|
|
else if (::connect (socket, (sockaddr *) &sin, sizeof (sin)) < 0) {
|
|
|
|
closesocket(socket);
|
2010-11-23 17:59:36 +03:00
|
|
|
socket = (SOCKET) -1;
|
2005-07-10 20:51:09 +04:00
|
|
|
BX_INFO(("com%d: connect() failed (host:%s, port:%d)",i+1, host, port));
|
|
|
|
}
|
2008-05-22 12:13:22 +04:00
|
|
|
|
2005-07-10 20:51:09 +04:00
|
|
|
BX_SER_THIS s[i].socket_id = socket;
|
2008-05-22 12:13:22 +04:00
|
|
|
if (socket > 0)
|
|
|
|
BX_INFO(("com%d - inet %s - socket_id: %d, ip:%s, port:%d",
|
|
|
|
i+1, server ? "server" : "client", socket, host, port));
|
|
|
|
} else if (!strncmp(mode, "pipe", 4)) {
|
|
|
|
if (strlen(dev) > 0) {
|
|
|
|
#ifdef WIN32
|
2008-12-21 11:56:26 +03:00
|
|
|
bx_bool server = !strcmp(mode, "pipe-server");
|
2008-05-22 12:13:22 +04:00
|
|
|
HANDLE pipe;
|
|
|
|
|
|
|
|
BX_SER_THIS s[i].io_mode = BX_SER_MODE_PIPE;
|
|
|
|
|
|
|
|
// server mode
|
|
|
|
if (server) {
|
|
|
|
pipe = CreateNamedPipe( dev,
|
|
|
|
PIPE_ACCESS_DUPLEX | FILE_FLAG_FIRST_PIPE_INSTANCE,
|
|
|
|
PIPE_TYPE_BYTE | PIPE_READMODE_BYTE | PIPE_WAIT,
|
|
|
|
1, 4096, 4096, 0, NULL);
|
|
|
|
|
|
|
|
if (pipe == INVALID_HANDLE_VALUE)
|
|
|
|
BX_PANIC(("com%d: CreateNamedPipe(%s) failed", i+1, dev));
|
|
|
|
|
|
|
|
BX_INFO(("com%d: waiting for client to connect to %s", i+1, dev));
|
|
|
|
if (!ConnectNamedPipe(pipe, NULL) && GetLastError() != ERROR_PIPE_CONNECTED)
|
|
|
|
{
|
|
|
|
CloseHandle(pipe);
|
|
|
|
pipe = INVALID_HANDLE_VALUE;
|
|
|
|
BX_PANIC(("com%d: ConnectNamedPipe(%s) failed", i+1, dev));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
// client mode
|
|
|
|
else {
|
|
|
|
pipe = CreateFile( dev,
|
|
|
|
GENERIC_READ | GENERIC_WRITE,
|
|
|
|
0, NULL, OPEN_EXISTING, 0, NULL);
|
|
|
|
|
|
|
|
if (pipe == INVALID_HANDLE_VALUE)
|
|
|
|
BX_INFO(("com%d: failed to open pipe %s", i+1, dev));
|
|
|
|
}
|
|
|
|
|
|
|
|
if (pipe != INVALID_HANDLE_VALUE)
|
|
|
|
BX_SER_THIS s[i].pipe = pipe;
|
|
|
|
#else
|
|
|
|
BX_PANIC(("support for serial mode '%s' not available", mode));
|
|
|
|
#endif
|
|
|
|
}
|
2004-07-28 23:36:42 +04:00
|
|
|
} else if (strcmp(mode, "null")) {
|
2008-05-22 12:13:22 +04:00
|
|
|
BX_PANIC(("unknown serial i/o mode '%s'", mode));
|
2004-07-28 23:36:42 +04:00
|
|
|
}
|
|
|
|
// simulate device connected
|
|
|
|
if (BX_SER_THIS s[i].io_mode != BX_SER_MODE_RAW) {
|
|
|
|
BX_SER_THIS s[i].modem_status.cts = 1;
|
|
|
|
BX_SER_THIS s[i].modem_status.dsr = 1;
|
|
|
|
}
|
2012-04-16 23:17:10 +04:00
|
|
|
count++;
|
2003-10-30 00:00:04 +03:00
|
|
|
BX_INFO(("com%d at 0x%04x irq %d", i+1, ports[i], BX_SER_THIS s[i].IRQ));
|
2002-10-06 23:04:47 +04:00
|
|
|
}
|
2001-04-10 05:04:59 +04:00
|
|
|
}
|
2012-04-16 23:17:10 +04:00
|
|
|
// Check if the device is disabled or not configured
|
|
|
|
if (count == 0) {
|
|
|
|
BX_INFO(("serial ports disabled"));
|
2012-07-08 11:01:25 +04:00
|
|
|
// mark unused plugin for removal
|
|
|
|
((bx_param_bool_c*)((bx_list_c*)SIM->get_param(BXPN_PLUGIN_CTRL))->get_by_name("serial"))->set(0);
|
2012-04-16 23:17:10 +04:00
|
|
|
return;
|
|
|
|
}
|
2009-03-03 23:34:50 +03:00
|
|
|
if ((BX_SER_THIS mouse_type == BX_MOUSE_TYPE_SERIAL) ||
|
|
|
|
(BX_SER_THIS mouse_type == BX_MOUSE_TYPE_SERIAL_WHEEL) ||
|
|
|
|
(BX_SER_THIS mouse_type == BX_MOUSE_TYPE_SERIAL_MSYS)) {
|
|
|
|
DEV_register_default_mouse(this, mouse_enq_static, NULL);
|
|
|
|
}
|
2001-04-10 05:04:59 +04:00
|
|
|
}
|
|
|
|
|
2006-05-27 19:54:49 +04:00
|
|
|
void bx_serial_c::register_state(void)
|
|
|
|
{
|
|
|
|
unsigned i, j;
|
|
|
|
char name[6];
|
|
|
|
bx_list_c *port;
|
|
|
|
|
2012-02-12 22:43:20 +04:00
|
|
|
bx_list_c *list = new bx_list_c(SIM->get_bochs_root(), "serial", "Serial Port State");
|
2006-05-27 19:54:49 +04:00
|
|
|
for (i=0; i<BX_N_SERIAL_PORTS; i++) {
|
|
|
|
sprintf(name, "%d", i);
|
2012-02-12 22:43:20 +04:00
|
|
|
port = new bx_list_c(list, name);
|
2006-05-27 19:54:49 +04:00
|
|
|
new bx_shadow_bool_c(port, "ls_interrupt", &BX_SER_THIS s[i].ls_interrupt);
|
|
|
|
new bx_shadow_bool_c(port, "ms_interrupt", &BX_SER_THIS s[i].ms_interrupt);
|
|
|
|
new bx_shadow_bool_c(port, "rx_interrupt", &BX_SER_THIS s[i].rx_interrupt);
|
|
|
|
new bx_shadow_bool_c(port, "tx_interrupt", &BX_SER_THIS s[i].tx_interrupt);
|
|
|
|
new bx_shadow_bool_c(port, "fifo_interrupt", &BX_SER_THIS s[i].fifo_interrupt);
|
|
|
|
new bx_shadow_bool_c(port, "ls_ipending", &BX_SER_THIS s[i].ls_ipending);
|
|
|
|
new bx_shadow_bool_c(port, "ms_ipending", &BX_SER_THIS s[i].ms_ipending);
|
|
|
|
new bx_shadow_bool_c(port, "rx_ipending", &BX_SER_THIS s[i].rx_ipending);
|
|
|
|
new bx_shadow_bool_c(port, "fifo_ipending", &BX_SER_THIS s[i].fifo_ipending);
|
|
|
|
new bx_shadow_num_c(port, "rx_fifo_end", &BX_SER_THIS s[i].rx_fifo_end);
|
|
|
|
new bx_shadow_num_c(port, "tx_fifo_end", &BX_SER_THIS s[i].tx_fifo_end);
|
|
|
|
new bx_shadow_num_c(port, "baudrate", &BX_SER_THIS s[i].baudrate);
|
|
|
|
new bx_shadow_num_c(port, "rx_pollstate", &BX_SER_THIS s[i].rx_pollstate);
|
|
|
|
new bx_shadow_num_c(port, "rxbuffer", &BX_SER_THIS s[i].rxbuffer, BASE_HEX);
|
|
|
|
new bx_shadow_num_c(port, "thrbuffer", &BX_SER_THIS s[i].thrbuffer, BASE_HEX);
|
2012-02-12 22:43:20 +04:00
|
|
|
bx_list_c *int_en = new bx_list_c(port, "int_enable");
|
2006-05-27 19:54:49 +04:00
|
|
|
new bx_shadow_bool_c(int_en, "rxdata_enable", &BX_SER_THIS s[i].int_enable.rxdata_enable);
|
|
|
|
new bx_shadow_bool_c(int_en, "txhold_enable", &BX_SER_THIS s[i].int_enable.txhold_enable);
|
|
|
|
new bx_shadow_bool_c(int_en, "rxlstat_enable", &BX_SER_THIS s[i].int_enable.rxlstat_enable);
|
|
|
|
new bx_shadow_bool_c(int_en, "modstat_enable", &BX_SER_THIS s[i].int_enable.modstat_enable);
|
2012-02-12 22:43:20 +04:00
|
|
|
bx_list_c *int_id = new bx_list_c(port, "int_ident");
|
2006-05-27 19:54:49 +04:00
|
|
|
new bx_shadow_bool_c(int_id, "ipending", &BX_SER_THIS s[i].int_ident.ipending);
|
|
|
|
new bx_shadow_num_c(int_id, "int_ID", &BX_SER_THIS s[i].int_ident.int_ID, BASE_HEX);
|
2012-02-12 22:43:20 +04:00
|
|
|
bx_list_c *fifo = new bx_list_c(port, "fifo_cntl");
|
2006-05-27 19:54:49 +04:00
|
|
|
new bx_shadow_bool_c(fifo, "enable", &BX_SER_THIS s[i].fifo_cntl.enable);
|
|
|
|
new bx_shadow_num_c(fifo, "rxtrigger", &BX_SER_THIS s[i].fifo_cntl.rxtrigger, BASE_HEX);
|
2012-02-12 22:43:20 +04:00
|
|
|
bx_list_c *lcntl = new bx_list_c(port, "line_cntl");
|
2006-05-27 19:54:49 +04:00
|
|
|
new bx_shadow_num_c(lcntl, "wordlen_sel", &BX_SER_THIS s[i].line_cntl.wordlen_sel, BASE_HEX);
|
|
|
|
new bx_shadow_bool_c(lcntl, "stopbits", &BX_SER_THIS s[i].line_cntl.stopbits);
|
|
|
|
new bx_shadow_bool_c(lcntl, "parity_enable", &BX_SER_THIS s[i].line_cntl.parity_enable);
|
|
|
|
new bx_shadow_bool_c(lcntl, "evenparity_sel", &BX_SER_THIS s[i].line_cntl.evenparity_sel);
|
|
|
|
new bx_shadow_bool_c(lcntl, "stick_parity", &BX_SER_THIS s[i].line_cntl.stick_parity);
|
|
|
|
new bx_shadow_bool_c(lcntl, "break_cntl", &BX_SER_THIS s[i].line_cntl.break_cntl);
|
|
|
|
new bx_shadow_bool_c(lcntl, "dlab", &BX_SER_THIS s[i].line_cntl.dlab);
|
2012-02-12 22:43:20 +04:00
|
|
|
bx_list_c *mcntl = new bx_list_c(port, "modem_cntl");
|
2006-05-27 19:54:49 +04:00
|
|
|
new bx_shadow_bool_c(mcntl, "dtr", &BX_SER_THIS s[i].modem_cntl.dtr);
|
|
|
|
new bx_shadow_bool_c(mcntl, "rts", &BX_SER_THIS s[i].modem_cntl.rts);
|
|
|
|
new bx_shadow_bool_c(mcntl, "out1", &BX_SER_THIS s[i].modem_cntl.out1);
|
|
|
|
new bx_shadow_bool_c(mcntl, "out2", &BX_SER_THIS s[i].modem_cntl.out2);
|
|
|
|
new bx_shadow_bool_c(mcntl, "local_loopback", &BX_SER_THIS s[i].modem_cntl.local_loopback);
|
2012-02-12 22:43:20 +04:00
|
|
|
bx_list_c *lstatus = new bx_list_c(port, "line_status");
|
2006-05-27 19:54:49 +04:00
|
|
|
new bx_shadow_bool_c(lstatus, "rxdata_ready", &BX_SER_THIS s[i].line_status.rxdata_ready);
|
|
|
|
new bx_shadow_bool_c(lstatus, "overrun_error", &BX_SER_THIS s[i].line_status.overrun_error);
|
|
|
|
new bx_shadow_bool_c(lstatus, "parity_error", &BX_SER_THIS s[i].line_status.parity_error);
|
|
|
|
new bx_shadow_bool_c(lstatus, "framing_error", &BX_SER_THIS s[i].line_status.framing_error);
|
|
|
|
new bx_shadow_bool_c(lstatus, "break_int", &BX_SER_THIS s[i].line_status.break_int);
|
|
|
|
new bx_shadow_bool_c(lstatus, "thr_empty", &BX_SER_THIS s[i].line_status.thr_empty);
|
|
|
|
new bx_shadow_bool_c(lstatus, "tsr_empty", &BX_SER_THIS s[i].line_status.tsr_empty);
|
|
|
|
new bx_shadow_bool_c(lstatus, "fifo_error", &BX_SER_THIS s[i].line_status.fifo_error);
|
2012-02-12 22:43:20 +04:00
|
|
|
bx_list_c *mstatus = new bx_list_c(port, "modem_status");
|
2006-05-27 19:54:49 +04:00
|
|
|
new bx_shadow_bool_c(mstatus, "delta_cts", &BX_SER_THIS s[i].modem_status.delta_cts);
|
|
|
|
new bx_shadow_bool_c(mstatus, "delta_dsr", &BX_SER_THIS s[i].modem_status.delta_dsr);
|
|
|
|
new bx_shadow_bool_c(mstatus, "ri_trailedge", &BX_SER_THIS s[i].modem_status.ri_trailedge);
|
|
|
|
new bx_shadow_bool_c(mstatus, "delta_dcd", &BX_SER_THIS s[i].modem_status.delta_dcd);
|
|
|
|
new bx_shadow_bool_c(mstatus, "cts", &BX_SER_THIS s[i].modem_status.cts);
|
|
|
|
new bx_shadow_bool_c(mstatus, "dsr", &BX_SER_THIS s[i].modem_status.dsr);
|
|
|
|
new bx_shadow_bool_c(mstatus, "ri", &BX_SER_THIS s[i].modem_status.ri);
|
|
|
|
new bx_shadow_bool_c(mstatus, "dcd", &BX_SER_THIS s[i].modem_status.dcd);
|
|
|
|
new bx_shadow_num_c(port, "scratch", &BX_SER_THIS s[i].scratch, BASE_HEX);
|
|
|
|
new bx_shadow_num_c(port, "tsrbuffer", &BX_SER_THIS s[i].tsrbuffer, BASE_HEX);
|
2012-02-12 22:43:20 +04:00
|
|
|
bx_list_c *rxfifo = new bx_list_c(port, "rx_fifo");
|
2006-05-27 19:54:49 +04:00
|
|
|
for (j=0; j<16; j++) {
|
|
|
|
sprintf(name, "0x%02x", j);
|
2006-05-30 02:33:38 +04:00
|
|
|
new bx_shadow_num_c(rxfifo, name, &BX_SER_THIS s[i].rx_fifo[j], BASE_HEX);
|
2006-05-27 19:54:49 +04:00
|
|
|
}
|
2012-02-12 22:43:20 +04:00
|
|
|
bx_list_c *txfifo = new bx_list_c(port, "tx_fifo");
|
2006-05-27 19:54:49 +04:00
|
|
|
for (j=0; j<16; j++) {
|
|
|
|
sprintf(name, "0x%02x", j);
|
2006-05-30 02:33:38 +04:00
|
|
|
new bx_shadow_num_c(txfifo, name, &BX_SER_THIS s[i].tx_fifo[j], BASE_HEX);
|
2006-05-27 19:54:49 +04:00
|
|
|
}
|
|
|
|
new bx_shadow_num_c(port, "divisor_lsb", &BX_SER_THIS s[i].divisor_lsb, BASE_HEX);
|
|
|
|
new bx_shadow_num_c(port, "divisor_msb", &BX_SER_THIS s[i].divisor_msb, BASE_HEX);
|
|
|
|
}
|
|
|
|
new bx_shadow_num_c(list, "detect_mouse", &BX_SER_THIS detect_mouse);
|
|
|
|
new bx_shadow_num_c(list, "mouse_delayed_dx", &BX_SER_THIS mouse_delayed_dx);
|
|
|
|
new bx_shadow_num_c(list, "mouse_delayed_dy", &BX_SER_THIS mouse_delayed_dy);
|
|
|
|
new bx_shadow_num_c(list, "mouse_delayed_dz", &BX_SER_THIS mouse_delayed_dz);
|
2012-02-12 22:43:20 +04:00
|
|
|
bx_list_c *mousebuf = new bx_list_c(list, "mouse_internal_buffer");
|
2006-05-27 19:54:49 +04:00
|
|
|
new bx_shadow_num_c(mousebuf, "num_elements", &BX_SER_THIS mouse_internal_buffer.num_elements);
|
2012-02-12 22:43:20 +04:00
|
|
|
bx_list_c *buffer = new bx_list_c(mousebuf, "buffer");
|
2006-05-27 19:54:49 +04:00
|
|
|
for (i=0; i<BX_MOUSE_BUFF_SIZE; i++) {
|
|
|
|
sprintf(name, "0x%02x", i);
|
2006-05-30 02:33:38 +04:00
|
|
|
new bx_shadow_num_c(buffer, name, &BX_SER_THIS mouse_internal_buffer.buffer[i], BASE_HEX);
|
2006-05-27 19:54:49 +04:00
|
|
|
}
|
|
|
|
new bx_shadow_num_c(mousebuf, "head", &BX_SER_THIS mouse_internal_buffer.head);
|
|
|
|
}
|
|
|
|
|
|
|
|
void bx_serial_c::lower_interrupt(Bit8u port)
|
2003-10-30 00:00:04 +03:00
|
|
|
{
|
|
|
|
/* If there are no more ints pending, clear the irq */
|
|
|
|
if ((BX_SER_THIS s[port].rx_interrupt == 0) &&
|
|
|
|
(BX_SER_THIS s[port].tx_interrupt == 0) &&
|
|
|
|
(BX_SER_THIS s[port].ls_interrupt == 0) &&
|
2003-11-09 03:14:43 +03:00
|
|
|
(BX_SER_THIS s[port].ms_interrupt == 0) &&
|
|
|
|
(BX_SER_THIS s[port].fifo_interrupt == 0)) {
|
2003-10-30 00:00:04 +03:00
|
|
|
DEV_pic_lower_irq(BX_SER_THIS s[port].IRQ);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-04-30 22:22:35 +04:00
|
|
|
void bx_serial_c::raise_interrupt(Bit8u port, int type)
|
2003-10-31 20:23:56 +03:00
|
|
|
{
|
|
|
|
bx_bool gen_int = 0;
|
|
|
|
|
|
|
|
switch (type) {
|
|
|
|
case BX_SER_INT_IER: /* IER has changed */
|
|
|
|
gen_int = 1;
|
|
|
|
break;
|
|
|
|
case BX_SER_INT_RXDATA:
|
|
|
|
if (BX_SER_THIS s[port].int_enable.rxdata_enable) {
|
|
|
|
BX_SER_THIS s[port].rx_interrupt = 1;
|
|
|
|
gen_int = 1;
|
|
|
|
} else {
|
|
|
|
BX_SER_THIS s[port].rx_ipending = 1;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case BX_SER_INT_TXHOLD:
|
|
|
|
if (BX_SER_THIS s[port].int_enable.txhold_enable) {
|
|
|
|
BX_SER_THIS s[port].tx_interrupt = 1;
|
|
|
|
gen_int = 1;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case BX_SER_INT_RXLSTAT:
|
|
|
|
if (BX_SER_THIS s[port].int_enable.rxlstat_enable) {
|
|
|
|
BX_SER_THIS s[port].ls_interrupt = 1;
|
|
|
|
gen_int = 1;
|
|
|
|
} else {
|
|
|
|
BX_SER_THIS s[port].ls_ipending = 1;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case BX_SER_INT_MODSTAT:
|
|
|
|
if ((BX_SER_THIS s[port].ms_ipending == 1) &&
|
|
|
|
(BX_SER_THIS s[port].int_enable.modstat_enable == 1)) {
|
|
|
|
BX_SER_THIS s[port].ms_interrupt = 1;
|
|
|
|
BX_SER_THIS s[port].ms_ipending = 0;
|
|
|
|
gen_int = 1;
|
|
|
|
}
|
|
|
|
break;
|
2003-11-09 03:14:43 +03:00
|
|
|
case BX_SER_INT_FIFO:
|
|
|
|
if (BX_SER_THIS s[port].int_enable.rxdata_enable) {
|
|
|
|
BX_SER_THIS s[port].fifo_interrupt = 1;
|
|
|
|
gen_int = 1;
|
|
|
|
} else {
|
|
|
|
BX_SER_THIS s[port].fifo_ipending = 1;
|
|
|
|
}
|
|
|
|
break;
|
2003-10-31 20:23:56 +03:00
|
|
|
}
|
|
|
|
if (gen_int && BX_SER_THIS s[port].modem_cntl.out2) {
|
|
|
|
DEV_pic_raise_irq(BX_SER_THIS s[port].IRQ);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2008-02-16 01:05:43 +03:00
|
|
|
// static IO port read callback handler
|
|
|
|
// redirects to non-static class handler to avoid virtual functions
|
2001-04-10 05:04:59 +04:00
|
|
|
|
2008-02-16 01:05:43 +03:00
|
|
|
Bit32u bx_serial_c::read_handler(void *this_ptr, Bit32u address, unsigned io_len)
|
2001-04-10 05:04:59 +04:00
|
|
|
{
|
|
|
|
#if !BX_USE_SER_SMF
|
|
|
|
bx_serial_c *class_ptr = (bx_serial_c *) this_ptr;
|
|
|
|
|
2008-02-16 01:05:43 +03:00
|
|
|
return class_ptr->read(address, io_len);
|
2001-04-10 05:04:59 +04:00
|
|
|
}
|
|
|
|
|
2008-02-16 01:05:43 +03:00
|
|
|
Bit32u bx_serial_c::read(Bit32u address, unsigned io_len)
|
2001-04-10 05:04:59 +04:00
|
|
|
{
|
|
|
|
#else
|
|
|
|
UNUSED(this_ptr);
|
|
|
|
#endif // !BX_USE_SER_SMF
|
2004-01-17 18:51:09 +03:00
|
|
|
Bit8u offset, val;
|
|
|
|
Bit8u port = 0;
|
2001-04-10 05:04:59 +04:00
|
|
|
|
2004-01-17 18:51:09 +03:00
|
|
|
offset = address & 0x07;
|
|
|
|
switch (address & 0x03f8) {
|
|
|
|
case 0x03f8: port = 0; break;
|
|
|
|
case 0x02f8: port = 1; break;
|
|
|
|
case 0x03e8: port = 2; break;
|
|
|
|
case 0x02e8: port = 3; break;
|
|
|
|
}
|
2001-04-10 05:04:59 +04:00
|
|
|
|
2004-01-17 18:51:09 +03:00
|
|
|
switch (offset) {
|
|
|
|
case BX_SER_RBR: /* receive buffer, or divisor latch LSB if DLAB set */
|
|
|
|
if (BX_SER_THIS s[port].line_cntl.dlab) {
|
|
|
|
val = BX_SER_THIS s[port].divisor_lsb;
|
2001-04-10 05:04:59 +04:00
|
|
|
} else {
|
2004-01-17 18:51:09 +03:00
|
|
|
if (BX_SER_THIS s[port].fifo_cntl.enable) {
|
|
|
|
val = BX_SER_THIS s[port].rx_fifo[0];
|
|
|
|
if (BX_SER_THIS s[port].rx_fifo_end > 0) {
|
|
|
|
memcpy(&BX_SER_THIS s[port].rx_fifo[0], &BX_SER_THIS s[port].rx_fifo[1], 15);
|
|
|
|
BX_SER_THIS s[port].rx_fifo_end--;
|
2003-11-09 03:14:43 +03:00
|
|
|
}
|
2004-01-17 18:51:09 +03:00
|
|
|
if (BX_SER_THIS s[port].rx_fifo_end == 0) {
|
|
|
|
BX_SER_THIS s[port].line_status.rxdata_ready = 0;
|
|
|
|
BX_SER_THIS s[port].rx_interrupt = 0;
|
|
|
|
BX_SER_THIS s[port].rx_ipending = 0;
|
|
|
|
BX_SER_THIS s[port].fifo_interrupt = 0;
|
|
|
|
BX_SER_THIS s[port].fifo_ipending = 0;
|
|
|
|
lower_interrupt(port);
|
2003-11-09 03:14:43 +03:00
|
|
|
}
|
|
|
|
} else {
|
2004-01-17 18:51:09 +03:00
|
|
|
val = BX_SER_THIS s[port].rxbuffer;
|
|
|
|
BX_SER_THIS s[port].line_status.rxdata_ready = 0;
|
|
|
|
BX_SER_THIS s[port].rx_interrupt = 0;
|
|
|
|
BX_SER_THIS s[port].rx_ipending = 0;
|
|
|
|
lower_interrupt(port);
|
2003-11-09 03:14:43 +03:00
|
|
|
}
|
2001-04-10 05:04:59 +04:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
2004-01-17 18:51:09 +03:00
|
|
|
case BX_SER_IER: /* interrupt enable register, or div. latch MSB */
|
|
|
|
if (BX_SER_THIS s[port].line_cntl.dlab) {
|
|
|
|
val = BX_SER_THIS s[port].divisor_msb;
|
2001-04-10 05:04:59 +04:00
|
|
|
} else {
|
2004-01-17 18:51:09 +03:00
|
|
|
val = BX_SER_THIS s[port].int_enable.rxdata_enable |
|
|
|
|
(BX_SER_THIS s[port].int_enable.txhold_enable << 1) |
|
|
|
|
(BX_SER_THIS s[port].int_enable.rxlstat_enable << 2) |
|
|
|
|
(BX_SER_THIS s[port].int_enable.modstat_enable << 3);
|
2001-04-10 05:04:59 +04:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
2004-01-17 18:51:09 +03:00
|
|
|
case BX_SER_IIR: /* interrupt ID register */
|
2001-04-10 05:04:59 +04:00
|
|
|
/*
|
|
|
|
* Set the interrupt ID based on interrupt source
|
|
|
|
*/
|
2004-01-17 18:51:09 +03:00
|
|
|
if (BX_SER_THIS s[port].ls_interrupt) {
|
|
|
|
BX_SER_THIS s[port].int_ident.int_ID = 0x3;
|
|
|
|
BX_SER_THIS s[port].int_ident.ipending = 0;
|
|
|
|
} else if (BX_SER_THIS s[port].fifo_interrupt) {
|
|
|
|
BX_SER_THIS s[port].int_ident.int_ID = 0x6;
|
|
|
|
BX_SER_THIS s[port].int_ident.ipending = 0;
|
|
|
|
} else if (BX_SER_THIS s[port].rx_interrupt) {
|
|
|
|
BX_SER_THIS s[port].int_ident.int_ID = 0x2;
|
|
|
|
BX_SER_THIS s[port].int_ident.ipending = 0;
|
|
|
|
} else if (BX_SER_THIS s[port].tx_interrupt) {
|
|
|
|
BX_SER_THIS s[port].int_ident.int_ID = 0x1;
|
|
|
|
BX_SER_THIS s[port].int_ident.ipending = 0;
|
|
|
|
} else if (BX_SER_THIS s[port].ms_interrupt) {
|
|
|
|
BX_SER_THIS s[port].int_ident.int_ID = 0x0;
|
|
|
|
BX_SER_THIS s[port].int_ident.ipending = 0;
|
2001-04-10 05:04:59 +04:00
|
|
|
} else {
|
2004-01-17 18:51:09 +03:00
|
|
|
BX_SER_THIS s[port].int_ident.int_ID = 0x0;
|
|
|
|
BX_SER_THIS s[port].int_ident.ipending = 1;
|
2002-01-20 19:35:32 +03:00
|
|
|
}
|
2004-01-17 18:51:09 +03:00
|
|
|
BX_SER_THIS s[port].tx_interrupt = 0;
|
|
|
|
lower_interrupt(port);
|
2001-04-10 05:04:59 +04:00
|
|
|
|
2004-01-17 18:51:09 +03:00
|
|
|
val = BX_SER_THIS s[port].int_ident.ipending |
|
|
|
|
(BX_SER_THIS s[port].int_ident.int_ID << 1) |
|
|
|
|
(BX_SER_THIS s[port].fifo_cntl.enable ? 0xc0 : 0x00);
|
2001-04-10 05:04:59 +04:00
|
|
|
break;
|
|
|
|
|
2004-01-17 18:51:09 +03:00
|
|
|
case BX_SER_LCR: /* Line control register */
|
2004-01-18 14:58:07 +03:00
|
|
|
val = BX_SER_THIS s[port].line_cntl.wordlen_sel |
|
|
|
|
(BX_SER_THIS s[port].line_cntl.stopbits << 2) |
|
|
|
|
(BX_SER_THIS s[port].line_cntl.parity_enable << 3) |
|
|
|
|
(BX_SER_THIS s[port].line_cntl.evenparity_sel << 4) |
|
|
|
|
(BX_SER_THIS s[port].line_cntl.stick_parity << 5) |
|
|
|
|
(BX_SER_THIS s[port].line_cntl.break_cntl << 6) |
|
|
|
|
(BX_SER_THIS s[port].line_cntl.dlab << 7);
|
2001-04-10 05:04:59 +04:00
|
|
|
break;
|
|
|
|
|
2004-01-17 18:51:09 +03:00
|
|
|
case BX_SER_MCR: /* MODEM control register */
|
|
|
|
val = BX_SER_THIS s[port].modem_cntl.dtr |
|
|
|
|
(BX_SER_THIS s[port].modem_cntl.rts << 1) |
|
|
|
|
(BX_SER_THIS s[port].modem_cntl.out1 << 2) |
|
|
|
|
(BX_SER_THIS s[port].modem_cntl.out2 << 3) |
|
|
|
|
(BX_SER_THIS s[port].modem_cntl.local_loopback << 4);
|
2001-04-10 05:04:59 +04:00
|
|
|
break;
|
|
|
|
|
2004-01-17 18:51:09 +03:00
|
|
|
case BX_SER_LSR: /* Line status register */
|
2004-01-18 14:58:07 +03:00
|
|
|
val = BX_SER_THIS s[port].line_status.rxdata_ready |
|
|
|
|
(BX_SER_THIS s[port].line_status.overrun_error << 1) |
|
|
|
|
(BX_SER_THIS s[port].line_status.parity_error << 2) |
|
|
|
|
(BX_SER_THIS s[port].line_status.framing_error << 3) |
|
|
|
|
(BX_SER_THIS s[port].line_status.break_int << 4) |
|
|
|
|
(BX_SER_THIS s[port].line_status.thr_empty << 5) |
|
|
|
|
(BX_SER_THIS s[port].line_status.tsr_empty << 6) |
|
|
|
|
(BX_SER_THIS s[port].line_status.fifo_error << 7);
|
2004-01-17 18:51:09 +03:00
|
|
|
BX_SER_THIS s[port].line_status.overrun_error = 0;
|
2004-03-10 00:58:37 +03:00
|
|
|
BX_SER_THIS s[port].line_status.framing_error = 0;
|
2004-01-17 18:51:09 +03:00
|
|
|
BX_SER_THIS s[port].line_status.break_int = 0;
|
|
|
|
BX_SER_THIS s[port].ls_interrupt = 0;
|
|
|
|
BX_SER_THIS s[port].ls_ipending = 0;
|
|
|
|
lower_interrupt(port);
|
2001-04-10 05:04:59 +04:00
|
|
|
break;
|
|
|
|
|
2004-01-17 18:51:09 +03:00
|
|
|
case BX_SER_MSR: /* MODEM status register */
|
2004-03-09 00:51:19 +03:00
|
|
|
#if USE_RAW_SERIAL
|
2011-04-30 22:22:35 +04:00
|
|
|
if (BX_SER_THIS s[port].io_mode == BX_SER_MODE_RAW) {
|
|
|
|
bx_bool prev_cts = BX_SER_THIS s[port].modem_status.cts;
|
|
|
|
bx_bool prev_dsr = BX_SER_THIS s[port].modem_status.dsr;
|
|
|
|
bx_bool prev_ri = BX_SER_THIS s[port].modem_status.ri;
|
|
|
|
bx_bool prev_dcd = BX_SER_THIS s[port].modem_status.dcd;
|
|
|
|
|
2004-07-28 23:36:42 +04:00
|
|
|
val = BX_SER_THIS s[port].raw->get_modem_status();
|
|
|
|
BX_SER_THIS s[port].modem_status.cts = (val & 0x10) >> 4;
|
|
|
|
BX_SER_THIS s[port].modem_status.dsr = (val & 0x20) >> 5;
|
|
|
|
BX_SER_THIS s[port].modem_status.ri = (val & 0x40) >> 6;
|
|
|
|
BX_SER_THIS s[port].modem_status.dcd = (val & 0x80) >> 7;
|
|
|
|
if (BX_SER_THIS s[port].modem_status.cts != prev_cts) {
|
|
|
|
BX_SER_THIS s[port].modem_status.delta_cts = 1;
|
|
|
|
}
|
|
|
|
if (BX_SER_THIS s[port].modem_status.dsr != prev_dsr) {
|
|
|
|
BX_SER_THIS s[port].modem_status.delta_dsr = 1;
|
|
|
|
}
|
|
|
|
if ((BX_SER_THIS s[port].modem_status.ri == 0) && (prev_ri == 1))
|
|
|
|
BX_SER_THIS s[port].modem_status.ri_trailedge = 1;
|
|
|
|
if (BX_SER_THIS s[port].modem_status.dcd != prev_dcd) {
|
|
|
|
BX_SER_THIS s[port].modem_status.delta_dcd = 1;
|
|
|
|
}
|
|
|
|
}
|
2011-04-30 22:22:35 +04:00
|
|
|
#endif
|
2004-01-18 14:58:07 +03:00
|
|
|
val = BX_SER_THIS s[port].modem_status.delta_cts |
|
|
|
|
(BX_SER_THIS s[port].modem_status.delta_dsr << 1) |
|
|
|
|
(BX_SER_THIS s[port].modem_status.ri_trailedge << 2) |
|
|
|
|
(BX_SER_THIS s[port].modem_status.delta_dcd << 3) |
|
|
|
|
(BX_SER_THIS s[port].modem_status.cts << 4) |
|
|
|
|
(BX_SER_THIS s[port].modem_status.dsr << 5) |
|
|
|
|
(BX_SER_THIS s[port].modem_status.ri << 6) |
|
|
|
|
(BX_SER_THIS s[port].modem_status.dcd << 7);
|
2004-01-17 18:51:09 +03:00
|
|
|
BX_SER_THIS s[port].modem_status.delta_cts = 0;
|
|
|
|
BX_SER_THIS s[port].modem_status.delta_dsr = 0;
|
|
|
|
BX_SER_THIS s[port].modem_status.ri_trailedge = 0;
|
|
|
|
BX_SER_THIS s[port].modem_status.delta_dcd = 0;
|
|
|
|
BX_SER_THIS s[port].ms_interrupt = 0;
|
|
|
|
BX_SER_THIS s[port].ms_ipending = 0;
|
|
|
|
lower_interrupt(port);
|
2001-04-10 05:04:59 +04:00
|
|
|
break;
|
|
|
|
|
2004-01-17 18:51:09 +03:00
|
|
|
case BX_SER_SCR: /* scratch register */
|
|
|
|
val = BX_SER_THIS s[port].scratch;
|
2001-04-10 05:04:59 +04:00
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
val = 0; // keep compiler happy
|
2004-01-17 18:51:09 +03:00
|
|
|
BX_PANIC(("unsupported io read from address=0x%04x!", address));
|
2001-04-10 05:04:59 +04:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2005-01-02 13:42:15 +03:00
|
|
|
BX_DEBUG(("com%d register read from address: 0x%04x = 0x%02x", port+1, address, val));
|
2001-04-10 05:04:59 +04:00
|
|
|
|
|
|
|
return(val);
|
|
|
|
}
|
|
|
|
|
2011-04-30 22:22:35 +04:00
|
|
|
// static IO port write callback handler
|
|
|
|
// redirects to non-static class handler to avoid virtual functions
|
2001-04-10 05:04:59 +04:00
|
|
|
|
2011-04-30 22:22:35 +04:00
|
|
|
void bx_serial_c::write_handler(void *this_ptr, Bit32u address, Bit32u value, unsigned io_len)
|
2001-04-10 05:04:59 +04:00
|
|
|
{
|
|
|
|
#if !BX_USE_SER_SMF
|
|
|
|
bx_serial_c *class_ptr = (bx_serial_c *) this_ptr;
|
|
|
|
|
|
|
|
class_ptr->write(address, value, io_len);
|
|
|
|
}
|
|
|
|
|
2011-04-30 22:22:35 +04:00
|
|
|
void bx_serial_c::write(Bit32u address, Bit32u value, unsigned io_len)
|
2001-04-10 05:04:59 +04:00
|
|
|
{
|
|
|
|
#else
|
|
|
|
UNUSED(this_ptr);
|
|
|
|
#endif // !BX_USE_SER_SMF
|
2002-10-25 15:44:41 +04:00
|
|
|
bx_bool gen_int = 0;
|
2004-03-10 00:58:37 +03:00
|
|
|
Bit8u offset, new_wordlen;
|
2004-02-28 16:10:57 +03:00
|
|
|
#if USE_RAW_SERIAL
|
2004-09-06 01:09:46 +04:00
|
|
|
bx_bool mcr_changed = 0;
|
2004-02-28 16:10:57 +03:00
|
|
|
Bit8u p_mode;
|
|
|
|
#endif
|
2004-01-17 18:51:09 +03:00
|
|
|
Bit8u port = 0;
|
2002-01-20 19:35:32 +03:00
|
|
|
|
2004-01-17 18:51:09 +03:00
|
|
|
offset = address & 0x07;
|
|
|
|
switch (address & 0x03f8) {
|
|
|
|
case 0x03f8: port = 0; break;
|
|
|
|
case 0x02f8: port = 1; break;
|
|
|
|
case 0x03e8: port = 2; break;
|
|
|
|
case 0x02e8: port = 3; break;
|
|
|
|
}
|
2005-01-02 13:42:15 +03:00
|
|
|
|
|
|
|
BX_DEBUG(("com%d register write to address: 0x%04x = 0x%02x", port+1, address, value));
|
|
|
|
|
2011-04-30 22:22:35 +04:00
|
|
|
bx_bool new_b0 = value & 0x01;
|
|
|
|
bx_bool new_b1 = (value & 0x02) >> 1;
|
|
|
|
bx_bool new_b2 = (value & 0x04) >> 2;
|
|
|
|
bx_bool new_b3 = (value & 0x08) >> 3;
|
|
|
|
bx_bool new_b4 = (value & 0x10) >> 4;
|
|
|
|
bx_bool new_b5 = (value & 0x20) >> 5;
|
|
|
|
bx_bool new_b6 = (value & 0x40) >> 6;
|
|
|
|
bx_bool new_b7 = (value & 0x80) >> 7;
|
2001-04-10 05:04:59 +04:00
|
|
|
|
2004-01-17 18:51:09 +03:00
|
|
|
switch (offset) {
|
|
|
|
case BX_SER_THR: /* transmit buffer, or divisor latch LSB if DLAB set */
|
|
|
|
if (BX_SER_THIS s[port].line_cntl.dlab) {
|
2004-01-18 14:58:07 +03:00
|
|
|
BX_SER_THIS s[port].divisor_lsb = value;
|
2001-04-10 05:04:59 +04:00
|
|
|
|
2004-01-17 18:51:09 +03:00
|
|
|
if ((value != 0) || (BX_SER_THIS s[port].divisor_msb != 0)) {
|
2004-01-18 14:58:07 +03:00
|
|
|
BX_SER_THIS s[port].baudrate = (int) (BX_PC_CLOCK_XTL /
|
|
|
|
(16 * ((BX_SER_THIS s[port].divisor_msb << 8) |
|
|
|
|
BX_SER_THIS s[port].divisor_lsb)));
|
|
|
|
}
|
2001-04-10 05:04:59 +04:00
|
|
|
} else {
|
2004-01-17 18:51:09 +03:00
|
|
|
Bit8u bitmask = 0xff >> (3 - BX_SER_THIS s[port].line_cntl.wordlen_sel);
|
|
|
|
if (BX_SER_THIS s[port].line_status.thr_empty) {
|
|
|
|
if (BX_SER_THIS s[port].fifo_cntl.enable) {
|
|
|
|
BX_SER_THIS s[port].tx_fifo[BX_SER_THIS s[port].tx_fifo_end++] = value & bitmask;
|
2003-11-09 03:14:43 +03:00
|
|
|
} else {
|
2004-01-17 18:51:09 +03:00
|
|
|
BX_SER_THIS s[port].thrbuffer = value & bitmask;
|
2003-11-09 03:14:43 +03:00
|
|
|
}
|
2004-01-17 18:51:09 +03:00
|
|
|
BX_SER_THIS s[port].line_status.thr_empty = 0;
|
|
|
|
if (BX_SER_THIS s[port].line_status.tsr_empty) {
|
|
|
|
if (BX_SER_THIS s[port].fifo_cntl.enable) {
|
|
|
|
BX_SER_THIS s[port].tsrbuffer = BX_SER_THIS s[port].tx_fifo[0];
|
|
|
|
memcpy(&BX_SER_THIS s[port].tx_fifo[0], &BX_SER_THIS s[port].tx_fifo[1], 15);
|
|
|
|
BX_SER_THIS s[port].line_status.thr_empty = (--BX_SER_THIS s[port].tx_fifo_end == 0);
|
2003-11-09 03:14:43 +03:00
|
|
|
} else {
|
2004-01-17 18:51:09 +03:00
|
|
|
BX_SER_THIS s[port].tsrbuffer = BX_SER_THIS s[port].thrbuffer;
|
|
|
|
BX_SER_THIS s[port].line_status.thr_empty = 1;
|
2003-11-09 03:14:43 +03:00
|
|
|
}
|
2004-01-17 18:51:09 +03:00
|
|
|
BX_SER_THIS s[port].line_status.tsr_empty = 0;
|
|
|
|
raise_interrupt(port, BX_SER_INT_TXHOLD);
|
|
|
|
bx_pc_system.activate_timer(BX_SER_THIS s[port].tx_timer_index,
|
|
|
|
(int) (1000000.0 / BX_SER_THIS s[port].baudrate *
|
|
|
|
(BX_SER_THIS s[port].line_cntl.wordlen_sel + 5)),
|
2003-09-15 00:16:25 +04:00
|
|
|
0); /* not continuous */
|
|
|
|
} else {
|
2004-01-17 18:51:09 +03:00
|
|
|
BX_SER_THIS s[port].tx_interrupt = 0;
|
|
|
|
lower_interrupt(port);
|
2003-09-15 00:16:25 +04:00
|
|
|
}
|
2002-01-20 19:35:32 +03:00
|
|
|
} else {
|
2004-01-17 18:51:09 +03:00
|
|
|
if (BX_SER_THIS s[port].fifo_cntl.enable) {
|
|
|
|
if (BX_SER_THIS s[port].tx_fifo_end < 16) {
|
|
|
|
BX_SER_THIS s[port].tx_fifo[BX_SER_THIS s[port].tx_fifo_end++] = value & bitmask;
|
2003-11-09 03:14:43 +03:00
|
|
|
} else {
|
2004-01-25 16:01:29 +03:00
|
|
|
BX_ERROR(("com%d: transmit FIFO overflow", port+1));
|
2003-11-09 03:14:43 +03:00
|
|
|
}
|
|
|
|
} else {
|
2004-01-25 16:01:29 +03:00
|
|
|
BX_ERROR(("com%d: write to tx hold register when not empty", port+1));
|
2003-11-09 03:14:43 +03:00
|
|
|
}
|
2002-01-20 19:35:32 +03:00
|
|
|
}
|
2001-04-10 05:04:59 +04:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
2004-01-17 18:51:09 +03:00
|
|
|
case BX_SER_IER: /* interrupt enable register, or div. latch MSB */
|
|
|
|
if (BX_SER_THIS s[port].line_cntl.dlab) {
|
2004-01-18 14:58:07 +03:00
|
|
|
BX_SER_THIS s[port].divisor_msb = value;
|
2001-04-10 05:04:59 +04:00
|
|
|
|
2004-01-17 18:51:09 +03:00
|
|
|
if ((value != 0) || (BX_SER_THIS s[port].divisor_lsb != 0)) {
|
2004-01-18 14:58:07 +03:00
|
|
|
BX_SER_THIS s[port].baudrate = (int) (BX_PC_CLOCK_XTL /
|
|
|
|
(16 * ((BX_SER_THIS s[port].divisor_msb << 8) |
|
|
|
|
BX_SER_THIS s[port].divisor_lsb)));
|
|
|
|
}
|
2001-04-10 05:04:59 +04:00
|
|
|
} else {
|
2004-03-10 00:58:37 +03:00
|
|
|
if (new_b3 != BX_SER_THIS s[port].int_enable.modstat_enable) {
|
|
|
|
BX_SER_THIS s[port].int_enable.modstat_enable = new_b3;
|
2004-01-17 18:51:09 +03:00
|
|
|
if (BX_SER_THIS s[port].int_enable.modstat_enable == 1) {
|
|
|
|
if (BX_SER_THIS s[port].ms_ipending == 1) {
|
|
|
|
BX_SER_THIS s[port].ms_interrupt = 1;
|
|
|
|
BX_SER_THIS s[port].ms_ipending = 0;
|
2003-10-30 00:00:04 +03:00
|
|
|
gen_int = 1;
|
|
|
|
}
|
|
|
|
} else {
|
2004-01-17 18:51:09 +03:00
|
|
|
if (BX_SER_THIS s[port].ms_interrupt == 1) {
|
|
|
|
BX_SER_THIS s[port].ms_interrupt = 0;
|
|
|
|
BX_SER_THIS s[port].ms_ipending = 1;
|
|
|
|
lower_interrupt(port);
|
2003-10-30 00:00:04 +03:00
|
|
|
}
|
2003-10-28 21:40:00 +03:00
|
|
|
}
|
2002-01-20 19:35:32 +03:00
|
|
|
}
|
2004-03-10 00:58:37 +03:00
|
|
|
if (new_b1 != BX_SER_THIS s[port].int_enable.txhold_enable) {
|
|
|
|
BX_SER_THIS s[port].int_enable.txhold_enable = new_b1;
|
2004-01-17 18:51:09 +03:00
|
|
|
if (BX_SER_THIS s[port].int_enable.txhold_enable == 1) {
|
|
|
|
BX_SER_THIS s[port].tx_interrupt = BX_SER_THIS s[port].line_status.thr_empty;
|
|
|
|
if (BX_SER_THIS s[port].tx_interrupt) gen_int = 1;
|
2003-10-28 21:40:00 +03:00
|
|
|
} else {
|
2004-01-17 18:51:09 +03:00
|
|
|
BX_SER_THIS s[port].tx_interrupt = 0;
|
|
|
|
lower_interrupt(port);
|
2003-10-28 21:40:00 +03:00
|
|
|
}
|
2002-01-20 19:35:32 +03:00
|
|
|
}
|
2004-03-10 00:58:37 +03:00
|
|
|
if (new_b0 != BX_SER_THIS s[port].int_enable.rxdata_enable) {
|
|
|
|
BX_SER_THIS s[port].int_enable.rxdata_enable = new_b0;
|
2004-01-17 18:51:09 +03:00
|
|
|
if (BX_SER_THIS s[port].int_enable.rxdata_enable == 1) {
|
|
|
|
if (BX_SER_THIS s[port].fifo_ipending == 1) {
|
|
|
|
BX_SER_THIS s[port].fifo_interrupt = 1;
|
|
|
|
BX_SER_THIS s[port].fifo_ipending = 0;
|
2003-11-09 03:14:43 +03:00
|
|
|
gen_int = 1;
|
|
|
|
}
|
2004-01-17 18:51:09 +03:00
|
|
|
if (BX_SER_THIS s[port].rx_ipending == 1) {
|
|
|
|
BX_SER_THIS s[port].rx_interrupt = 1;
|
|
|
|
BX_SER_THIS s[port].rx_ipending = 0;
|
2003-10-30 00:00:04 +03:00
|
|
|
gen_int = 1;
|
|
|
|
}
|
|
|
|
} else {
|
2004-01-17 18:51:09 +03:00
|
|
|
if (BX_SER_THIS s[port].rx_interrupt == 1) {
|
|
|
|
BX_SER_THIS s[port].rx_interrupt = 0;
|
|
|
|
BX_SER_THIS s[port].rx_ipending = 1;
|
|
|
|
lower_interrupt(port);
|
2003-10-30 00:00:04 +03:00
|
|
|
}
|
2004-01-17 18:51:09 +03:00
|
|
|
if (BX_SER_THIS s[port].fifo_interrupt == 1) {
|
|
|
|
BX_SER_THIS s[port].fifo_interrupt = 0;
|
|
|
|
BX_SER_THIS s[port].fifo_ipending = 1;
|
|
|
|
lower_interrupt(port);
|
2003-11-09 03:14:43 +03:00
|
|
|
}
|
2003-10-28 21:40:00 +03:00
|
|
|
}
|
2002-01-20 19:35:32 +03:00
|
|
|
}
|
2004-03-10 00:58:37 +03:00
|
|
|
if (new_b2 != BX_SER_THIS s[port].int_enable.rxlstat_enable) {
|
|
|
|
BX_SER_THIS s[port].int_enable.rxlstat_enable = new_b2;
|
2004-01-17 18:51:09 +03:00
|
|
|
if (BX_SER_THIS s[port].int_enable.rxlstat_enable == 1) {
|
|
|
|
if (BX_SER_THIS s[port].ls_ipending == 1) {
|
|
|
|
BX_SER_THIS s[port].ls_interrupt = 1;
|
|
|
|
BX_SER_THIS s[port].ls_ipending = 0;
|
2003-10-30 00:00:04 +03:00
|
|
|
gen_int = 1;
|
|
|
|
}
|
|
|
|
} else {
|
2004-01-17 18:51:09 +03:00
|
|
|
if (BX_SER_THIS s[port].ls_interrupt == 1) {
|
|
|
|
BX_SER_THIS s[port].ls_interrupt = 0;
|
|
|
|
BX_SER_THIS s[port].ls_ipending = 1;
|
|
|
|
lower_interrupt(port);
|
2003-10-30 00:00:04 +03:00
|
|
|
}
|
2003-10-28 21:40:00 +03:00
|
|
|
}
|
2002-01-20 19:35:32 +03:00
|
|
|
}
|
2004-01-17 18:51:09 +03:00
|
|
|
if (gen_int) raise_interrupt(port, BX_SER_INT_IER);
|
2001-04-10 05:04:59 +04:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
2004-01-17 18:51:09 +03:00
|
|
|
case BX_SER_FCR: /* FIFO control register */
|
2004-03-10 00:58:37 +03:00
|
|
|
if (new_b0 && !BX_SER_THIS s[port].fifo_cntl.enable) {
|
2004-01-25 16:01:29 +03:00
|
|
|
BX_INFO(("com%d: FIFO enabled", port+1));
|
2004-01-17 18:51:09 +03:00
|
|
|
BX_SER_THIS s[port].rx_fifo_end = 0;
|
|
|
|
BX_SER_THIS s[port].tx_fifo_end = 0;
|
2003-11-09 03:14:43 +03:00
|
|
|
}
|
2004-03-10 00:58:37 +03:00
|
|
|
BX_SER_THIS s[port].fifo_cntl.enable = new_b0;
|
|
|
|
if (new_b1) {
|
2004-01-17 18:51:09 +03:00
|
|
|
BX_SER_THIS s[port].rx_fifo_end = 0;
|
2003-11-09 03:14:43 +03:00
|
|
|
}
|
2004-03-10 00:58:37 +03:00
|
|
|
if (new_b2) {
|
2004-01-17 18:51:09 +03:00
|
|
|
BX_SER_THIS s[port].tx_fifo_end = 0;
|
2003-11-09 03:14:43 +03:00
|
|
|
}
|
2004-01-17 18:51:09 +03:00
|
|
|
BX_SER_THIS s[port].fifo_cntl.rxtrigger = (value & 0xc0) >> 6;
|
2001-04-10 05:04:59 +04:00
|
|
|
break;
|
|
|
|
|
2004-01-17 18:51:09 +03:00
|
|
|
case BX_SER_LCR: /* Line control register */
|
2004-02-28 16:10:57 +03:00
|
|
|
new_wordlen = value & 0x03;
|
|
|
|
#if USE_RAW_SERIAL
|
2011-04-30 22:22:35 +04:00
|
|
|
if (BX_SER_THIS s[port].io_mode == BX_SER_MODE_RAW) {
|
2004-07-28 23:36:42 +04:00
|
|
|
if (BX_SER_THIS s[port].line_cntl.wordlen_sel != new_wordlen) {
|
|
|
|
BX_SER_THIS s[port].raw->set_data_bits(new_wordlen + 5);
|
|
|
|
}
|
|
|
|
if (new_b2 != BX_SER_THIS s[port].line_cntl.stopbits) {
|
|
|
|
BX_SER_THIS s[port].raw->set_stop_bits(new_b2 ? 2 : 1);
|
|
|
|
}
|
|
|
|
if ((new_b3 != BX_SER_THIS s[port].line_cntl.parity_enable) ||
|
|
|
|
(new_b4 != BX_SER_THIS s[port].line_cntl.evenparity_sel) ||
|
|
|
|
(new_b5 != BX_SER_THIS s[port].line_cntl.stick_parity)) {
|
|
|
|
if (new_b3 == 0) {
|
|
|
|
p_mode = P_NONE;
|
|
|
|
} else {
|
|
|
|
p_mode = ((value & 0x30) >> 4) + 1;
|
|
|
|
}
|
|
|
|
BX_SER_THIS s[port].raw->set_parity_mode(p_mode);
|
|
|
|
}
|
|
|
|
if ((new_b6 != BX_SER_THIS s[port].line_cntl.break_cntl) &&
|
|
|
|
(!BX_SER_THIS s[port].modem_cntl.local_loopback)) {
|
|
|
|
BX_SER_THIS s[port].raw->set_break(new_b6);
|
2004-02-28 16:10:57 +03:00
|
|
|
}
|
2004-07-28 23:36:42 +04:00
|
|
|
}
|
2011-04-30 22:22:35 +04:00
|
|
|
#endif // USE_RAW_SERIAL
|
2004-02-28 16:10:57 +03:00
|
|
|
BX_SER_THIS s[port].line_cntl.wordlen_sel = new_wordlen;
|
2001-04-10 05:04:59 +04:00
|
|
|
/* These are ignored, but set them up so they can be read back */
|
2004-03-10 00:58:37 +03:00
|
|
|
BX_SER_THIS s[port].line_cntl.stopbits = new_b2;
|
|
|
|
BX_SER_THIS s[port].line_cntl.parity_enable = new_b3;
|
|
|
|
BX_SER_THIS s[port].line_cntl.evenparity_sel = new_b4;
|
|
|
|
BX_SER_THIS s[port].line_cntl.stick_parity = new_b5;
|
|
|
|
BX_SER_THIS s[port].line_cntl.break_cntl = new_b6;
|
|
|
|
if (BX_SER_THIS s[port].modem_cntl.local_loopback &&
|
|
|
|
BX_SER_THIS s[port].line_cntl.break_cntl) {
|
|
|
|
BX_SER_THIS s[port].line_status.break_int = 1;
|
|
|
|
BX_SER_THIS s[port].line_status.framing_error = 1;
|
|
|
|
rx_fifo_enq(port, 0x00);
|
|
|
|
}
|
2001-04-10 05:04:59 +04:00
|
|
|
/* used when doing future writes */
|
2004-03-10 00:58:37 +03:00
|
|
|
if (!new_b7 && BX_SER_THIS s[port].line_cntl.dlab) {
|
2004-01-18 14:58:07 +03:00
|
|
|
// Start the receive polling process if not already started
|
|
|
|
// and there is a valid baudrate.
|
|
|
|
if (BX_SER_THIS s[port].rx_pollstate == BX_SER_RXIDLE &&
|
|
|
|
BX_SER_THIS s[port].baudrate != 0) {
|
|
|
|
BX_SER_THIS s[port].rx_pollstate = BX_SER_RXPOLL;
|
|
|
|
bx_pc_system.activate_timer(BX_SER_THIS s[port].rx_timer_index,
|
|
|
|
(int) (1000000.0 / BX_SER_THIS s[port].baudrate *
|
|
|
|
(BX_SER_THIS s[port].line_cntl.wordlen_sel + 5)),
|
|
|
|
0); /* not continuous */
|
|
|
|
}
|
2004-02-28 16:10:57 +03:00
|
|
|
#if USE_RAW_SERIAL
|
2011-04-30 22:22:35 +04:00
|
|
|
if (BX_SER_THIS s[port].io_mode == BX_SER_MODE_RAW) {
|
2004-07-28 23:36:42 +04:00
|
|
|
BX_SER_THIS s[port].raw->set_baudrate(BX_SER_THIS s[port].baudrate);
|
|
|
|
}
|
2011-04-30 22:22:35 +04:00
|
|
|
#endif // USE_RAW_SERIAL
|
2004-01-25 16:01:29 +03:00
|
|
|
BX_DEBUG(("com%d: baud rate set - %d", port+1, BX_SER_THIS s[port].baudrate));
|
2001-04-10 05:04:59 +04:00
|
|
|
}
|
2004-03-10 00:58:37 +03:00
|
|
|
BX_SER_THIS s[port].line_cntl.dlab = new_b7;
|
2001-04-10 05:04:59 +04:00
|
|
|
break;
|
2002-01-20 19:35:32 +03:00
|
|
|
|
2004-01-17 18:51:09 +03:00
|
|
|
case BX_SER_MCR: /* MODEM control register */
|
2004-12-03 00:34:26 +03:00
|
|
|
if ((BX_SER_THIS s[port].io_mode == BX_SER_MODE_MOUSE) &&
|
2006-12-31 14:56:14 +03:00
|
|
|
((BX_SER_THIS s[port].line_cntl.wordlen_sel == 2) ||
|
|
|
|
(BX_SER_THIS s[port].line_cntl.wordlen_sel == 3))) {
|
2004-12-05 23:23:39 +03:00
|
|
|
if (new_b0 && !new_b1) BX_SER_THIS detect_mouse = 1;
|
|
|
|
if (new_b0 && new_b1 && (BX_SER_THIS detect_mouse == 1)) BX_SER_THIS detect_mouse = 2;
|
2004-11-27 13:09:41 +03:00
|
|
|
}
|
2004-03-10 00:58:37 +03:00
|
|
|
#if USE_RAW_SERIAL
|
2011-04-30 22:22:35 +04:00
|
|
|
if (BX_SER_THIS s[port].io_mode == BX_SER_MODE_RAW) {
|
2004-07-28 23:36:42 +04:00
|
|
|
mcr_changed = (BX_SER_THIS s[port].modem_cntl.dtr != new_b0) |
|
|
|
|
(BX_SER_THIS s[port].modem_cntl.rts != new_b1);
|
|
|
|
}
|
2011-04-30 22:22:35 +04:00
|
|
|
#endif
|
2004-03-10 00:58:37 +03:00
|
|
|
BX_SER_THIS s[port].modem_cntl.dtr = new_b0;
|
|
|
|
BX_SER_THIS s[port].modem_cntl.rts = new_b1;
|
|
|
|
BX_SER_THIS s[port].modem_cntl.out1 = new_b2;
|
|
|
|
BX_SER_THIS s[port].modem_cntl.out2 = new_b3;
|
|
|
|
|
2004-03-10 01:17:33 +03:00
|
|
|
if (new_b4 != BX_SER_THIS s[port].modem_cntl.local_loopback) {
|
2004-03-10 00:58:37 +03:00
|
|
|
BX_SER_THIS s[port].modem_cntl.local_loopback = new_b4;
|
|
|
|
if (BX_SER_THIS s[port].modem_cntl.local_loopback) {
|
|
|
|
/* transition to loopback mode */
|
|
|
|
#if USE_RAW_SERIAL
|
2011-04-30 22:22:35 +04:00
|
|
|
if (BX_SER_THIS s[port].io_mode == BX_SER_MODE_RAW) {
|
2004-07-28 23:36:42 +04:00
|
|
|
if (BX_SER_THIS s[port].modem_cntl.dtr ||
|
|
|
|
BX_SER_THIS s[port].modem_cntl.rts) {
|
|
|
|
BX_SER_THIS s[port].raw->set_modem_control(0);
|
|
|
|
}
|
|
|
|
}
|
2011-04-30 22:22:35 +04:00
|
|
|
#endif
|
2004-03-10 00:58:37 +03:00
|
|
|
if (BX_SER_THIS s[port].line_cntl.break_cntl) {
|
|
|
|
#if USE_RAW_SERIAL
|
2011-04-30 22:22:35 +04:00
|
|
|
if (BX_SER_THIS s[port].io_mode == BX_SER_MODE_RAW) {
|
2004-07-28 23:36:42 +04:00
|
|
|
BX_SER_THIS s[port].raw->set_break(0);
|
|
|
|
}
|
2011-04-30 22:22:35 +04:00
|
|
|
#endif
|
2004-03-10 00:58:37 +03:00
|
|
|
BX_SER_THIS s[port].line_status.break_int = 1;
|
|
|
|
BX_SER_THIS s[port].line_status.framing_error = 1;
|
|
|
|
rx_fifo_enq(port, 0x00);
|
|
|
|
}
|
2011-04-30 22:22:35 +04:00
|
|
|
}
|
|
|
|
else {
|
2004-03-10 00:58:37 +03:00
|
|
|
/* transition to normal mode */
|
|
|
|
#if USE_RAW_SERIAL
|
2011-04-30 22:22:35 +04:00
|
|
|
if (BX_SER_THIS s[port].io_mode == BX_SER_MODE_RAW) {
|
2004-07-28 23:36:42 +04:00
|
|
|
mcr_changed = 1;
|
|
|
|
if (BX_SER_THIS s[port].line_cntl.break_cntl) {
|
|
|
|
BX_SER_THIS s[port].raw->set_break(0);
|
|
|
|
}
|
|
|
|
}
|
2011-04-30 22:22:35 +04:00
|
|
|
#endif
|
2004-03-10 00:58:37 +03:00
|
|
|
}
|
|
|
|
}
|
2004-01-17 18:51:09 +03:00
|
|
|
|
|
|
|
if (BX_SER_THIS s[port].modem_cntl.local_loopback) {
|
2011-04-30 22:22:35 +04:00
|
|
|
bx_bool prev_cts = BX_SER_THIS s[port].modem_status.cts;
|
|
|
|
bx_bool prev_dsr = BX_SER_THIS s[port].modem_status.dsr;
|
|
|
|
bx_bool prev_ri = BX_SER_THIS s[port].modem_status.ri;
|
|
|
|
bx_bool prev_dcd = BX_SER_THIS s[port].modem_status.dcd;
|
2004-01-17 18:51:09 +03:00
|
|
|
BX_SER_THIS s[port].modem_status.cts = BX_SER_THIS s[port].modem_cntl.rts;
|
|
|
|
BX_SER_THIS s[port].modem_status.dsr = BX_SER_THIS s[port].modem_cntl.dtr;
|
|
|
|
BX_SER_THIS s[port].modem_status.ri = BX_SER_THIS s[port].modem_cntl.out1;
|
|
|
|
BX_SER_THIS s[port].modem_status.dcd = BX_SER_THIS s[port].modem_cntl.out2;
|
|
|
|
if (BX_SER_THIS s[port].modem_status.cts != prev_cts) {
|
|
|
|
BX_SER_THIS s[port].modem_status.delta_cts = 1;
|
|
|
|
BX_SER_THIS s[port].ms_ipending = 1;
|
2002-01-20 19:35:32 +03:00
|
|
|
}
|
2004-01-17 18:51:09 +03:00
|
|
|
if (BX_SER_THIS s[port].modem_status.dsr != prev_dsr) {
|
|
|
|
BX_SER_THIS s[port].modem_status.delta_dsr = 1;
|
|
|
|
BX_SER_THIS s[port].ms_ipending = 1;
|
2002-01-20 19:35:32 +03:00
|
|
|
}
|
2004-01-17 18:51:09 +03:00
|
|
|
if (BX_SER_THIS s[port].modem_status.ri != prev_ri)
|
|
|
|
BX_SER_THIS s[port].ms_ipending = 1;
|
|
|
|
if ((BX_SER_THIS s[port].modem_status.ri == 0) && (prev_ri == 1))
|
|
|
|
BX_SER_THIS s[port].modem_status.ri_trailedge = 1;
|
|
|
|
if (BX_SER_THIS s[port].modem_status.dcd != prev_dcd) {
|
|
|
|
BX_SER_THIS s[port].modem_status.delta_dcd = 1;
|
|
|
|
BX_SER_THIS s[port].ms_ipending = 1;
|
2002-01-20 19:35:32 +03:00
|
|
|
}
|
2004-01-17 18:51:09 +03:00
|
|
|
raise_interrupt(port, BX_SER_INT_MODSTAT);
|
2001-04-10 05:04:59 +04:00
|
|
|
} else {
|
2004-11-27 13:09:41 +03:00
|
|
|
if (BX_SER_THIS s[port].io_mode == BX_SER_MODE_MOUSE) {
|
2004-12-05 23:23:39 +03:00
|
|
|
if (BX_SER_THIS detect_mouse == 2) {
|
2006-12-31 14:56:14 +03:00
|
|
|
if ((BX_SER_THIS mouse_type == BX_MOUSE_TYPE_SERIAL) ||
|
|
|
|
(BX_SER_THIS mouse_type == BX_MOUSE_TYPE_SERIAL_MSYS)) {
|
2004-12-08 00:06:35 +03:00
|
|
|
BX_SER_THIS mouse_internal_buffer.head = 0;
|
2004-12-05 23:23:39 +03:00
|
|
|
BX_SER_THIS mouse_internal_buffer.num_elements = 1;
|
|
|
|
BX_SER_THIS mouse_internal_buffer.buffer[0] = 'M';
|
2006-12-31 14:56:14 +03:00
|
|
|
} else if (BX_SER_THIS mouse_type == BX_MOUSE_TYPE_SERIAL_WHEEL) {
|
2004-12-08 00:06:35 +03:00
|
|
|
BX_SER_THIS mouse_internal_buffer.head = 0;
|
2004-12-05 23:23:39 +03:00
|
|
|
BX_SER_THIS mouse_internal_buffer.num_elements = 6;
|
|
|
|
BX_SER_THIS mouse_internal_buffer.buffer[0] = 'M';
|
|
|
|
BX_SER_THIS mouse_internal_buffer.buffer[1] = 'Z';
|
|
|
|
BX_SER_THIS mouse_internal_buffer.buffer[2] = '@';
|
|
|
|
BX_SER_THIS mouse_internal_buffer.buffer[3] = '\0';
|
|
|
|
BX_SER_THIS mouse_internal_buffer.buffer[4] = '\0';
|
|
|
|
BX_SER_THIS mouse_internal_buffer.buffer[5] = '\0';
|
|
|
|
}
|
|
|
|
BX_SER_THIS detect_mouse = 0;
|
|
|
|
}
|
2004-11-27 13:09:41 +03:00
|
|
|
}
|
2004-12-05 23:23:39 +03:00
|
|
|
|
2004-07-28 23:36:42 +04:00
|
|
|
if (BX_SER_THIS s[port].io_mode == BX_SER_MODE_RAW) {
|
2004-03-09 00:51:19 +03:00
|
|
|
#if USE_RAW_SERIAL
|
2004-07-28 23:36:42 +04:00
|
|
|
if (mcr_changed) {
|
|
|
|
BX_SER_THIS s[port].raw->set_modem_control(value & 0x03);
|
|
|
|
}
|
2004-03-09 00:51:19 +03:00
|
|
|
#endif
|
2011-04-30 22:22:35 +04:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2004-07-28 23:36:42 +04:00
|
|
|
/* simulate device connected */
|
|
|
|
BX_SER_THIS s[port].modem_status.cts = 1;
|
|
|
|
BX_SER_THIS s[port].modem_status.dsr = 1;
|
|
|
|
BX_SER_THIS s[port].modem_status.ri = 0;
|
|
|
|
BX_SER_THIS s[port].modem_status.dcd = 0;
|
|
|
|
}
|
2001-04-10 05:04:59 +04:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
2004-01-17 18:51:09 +03:00
|
|
|
case BX_SER_LSR: /* Line status register */
|
2004-01-25 16:01:29 +03:00
|
|
|
BX_ERROR(("com%d: write to line status register ignored", port+1));
|
2001-04-10 05:04:59 +04:00
|
|
|
break;
|
|
|
|
|
2004-01-17 18:51:09 +03:00
|
|
|
case BX_SER_MSR: /* MODEM status register */
|
2004-01-25 16:01:29 +03:00
|
|
|
BX_ERROR(("com%d: write to MODEM status register ignored", port+1));
|
2001-04-10 05:04:59 +04:00
|
|
|
break;
|
|
|
|
|
2004-01-17 18:51:09 +03:00
|
|
|
case BX_SER_SCR: /* scratch register */
|
|
|
|
BX_SER_THIS s[port].scratch = value;
|
2001-04-10 05:04:59 +04:00
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
2002-08-27 21:24:36 +04:00
|
|
|
BX_PANIC(("unsupported io write to address=0x%04x, value = 0x%02x!",
|
merge in BRANCH-io-cleanup.
To see the commit logs for this use either cvsweb or
cvs update -r BRANCH-io-cleanup and then 'cvs log' the various files.
In general this provides a generic interface for logging.
logfunctions:: is a class that is inherited by some classes, and also
. allocated as a standalone global called 'genlog'. All logging uses
. one of the ::info(), ::error(), ::ldebug(), ::panic() methods of this
. class through 'BX_INFO(), BX_ERROR(), BX_DEBUG(), BX_PANIC()' macros
. respectively.
.
. An example usage:
. BX_INFO(("Hello, World!\n"));
iofunctions:: is a class that is allocated once by default, and assigned
as the iofunction of each logfunctions instance. It is this class that
maintains the file descriptor and other output related code, at this
point using vfprintf(). At some future point, someone may choose to
write a gui 'console' for bochs to which messages would be redirected
simply by assigning a different iofunction class to the various logfunctions
objects.
More cleanup is coming, but this works for now. If you want to see alot
of debugging output, in main.cc, change onoff[LOGLEV_DEBUG]=0 to =1.
Comments, bugs, flames, to me: todd@fries.net
2001-05-15 18:49:57 +04:00
|
|
|
(unsigned) address, (unsigned) value));
|
2001-04-10 05:04:59 +04:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-04-30 22:22:35 +04:00
|
|
|
void bx_serial_c::rx_fifo_enq(Bit8u port, Bit8u data)
|
2003-11-09 03:14:43 +03:00
|
|
|
{
|
|
|
|
bx_bool gen_int = 0;
|
|
|
|
|
|
|
|
if (BX_SER_THIS s[port].fifo_cntl.enable) {
|
|
|
|
if (BX_SER_THIS s[port].rx_fifo_end == 16) {
|
2004-01-25 16:01:29 +03:00
|
|
|
BX_ERROR(("com%d: receive FIFO overflow", port+1));
|
2003-11-09 03:14:43 +03:00
|
|
|
BX_SER_THIS s[port].line_status.overrun_error = 1;
|
|
|
|
raise_interrupt(port, BX_SER_INT_RXLSTAT);
|
|
|
|
} else {
|
2004-01-17 18:51:09 +03:00
|
|
|
BX_SER_THIS s[port].rx_fifo[BX_SER_THIS s[port].rx_fifo_end++] = data;
|
2003-11-09 03:14:43 +03:00
|
|
|
switch (BX_SER_THIS s[port].fifo_cntl.rxtrigger) {
|
|
|
|
case 1:
|
2004-01-17 18:51:09 +03:00
|
|
|
if (BX_SER_THIS s[port].rx_fifo_end == 4) gen_int = 1;
|
2003-11-09 03:14:43 +03:00
|
|
|
break;
|
|
|
|
case 2:
|
2004-01-17 18:51:09 +03:00
|
|
|
if (BX_SER_THIS s[port].rx_fifo_end == 8) gen_int = 1;
|
2003-11-09 03:14:43 +03:00
|
|
|
break;
|
|
|
|
case 3:
|
2004-01-17 18:51:09 +03:00
|
|
|
if (BX_SER_THIS s[port].rx_fifo_end == 14) gen_int = 1;
|
2003-11-09 03:14:43 +03:00
|
|
|
break;
|
|
|
|
default:
|
|
|
|
gen_int = 1;
|
|
|
|
}
|
|
|
|
if (gen_int) {
|
2004-01-17 18:51:09 +03:00
|
|
|
bx_pc_system.deactivate_timer(BX_SER_THIS s[port].fifo_timer_index);
|
2003-11-09 03:14:43 +03:00
|
|
|
BX_SER_THIS s[port].line_status.rxdata_ready = 1;
|
|
|
|
raise_interrupt(port, BX_SER_INT_RXDATA);
|
|
|
|
} else {
|
2004-01-17 18:51:09 +03:00
|
|
|
bx_pc_system.activate_timer(BX_SER_THIS s[port].fifo_timer_index,
|
|
|
|
(int) (1000000.0 / BX_SER_THIS s[port].baudrate *
|
|
|
|
(BX_SER_THIS s[port].line_cntl.wordlen_sel + 5) * 16),
|
2003-11-09 03:14:43 +03:00
|
|
|
0); /* not continuous */
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
if (BX_SER_THIS s[port].line_status.rxdata_ready == 1) {
|
2004-01-25 16:01:29 +03:00
|
|
|
BX_ERROR(("com%d: overrun error", port+1));
|
2003-11-09 03:14:43 +03:00
|
|
|
BX_SER_THIS s[port].line_status.overrun_error = 1;
|
|
|
|
raise_interrupt(port, BX_SER_INT_RXLSTAT);
|
|
|
|
}
|
|
|
|
BX_SER_THIS s[port].rxbuffer = data;
|
|
|
|
BX_SER_THIS s[port].line_status.rxdata_ready = 1;
|
|
|
|
raise_interrupt(port, BX_SER_INT_RXDATA);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-04-30 22:22:35 +04:00
|
|
|
void bx_serial_c::tx_timer_handler(void *this_ptr)
|
2001-04-10 05:04:59 +04:00
|
|
|
{
|
|
|
|
bx_serial_c *class_ptr = (bx_serial_c *) this_ptr;
|
|
|
|
|
|
|
|
class_ptr->tx_timer();
|
|
|
|
}
|
|
|
|
|
2011-04-30 22:22:35 +04:00
|
|
|
void bx_serial_c::tx_timer(void)
|
2001-04-10 05:04:59 +04:00
|
|
|
{
|
2003-11-09 03:14:43 +03:00
|
|
|
bx_bool gen_int = 0;
|
2004-01-17 18:51:09 +03:00
|
|
|
Bit8u port = 0;
|
|
|
|
int timer_id;
|
|
|
|
|
|
|
|
timer_id = bx_pc_system.triggeredTimerID();
|
|
|
|
if (timer_id == BX_SER_THIS s[0].tx_timer_index) {
|
|
|
|
port = 0;
|
|
|
|
} else if (timer_id == BX_SER_THIS s[1].tx_timer_index) {
|
|
|
|
port = 1;
|
|
|
|
} else if (timer_id == BX_SER_THIS s[2].tx_timer_index) {
|
|
|
|
port = 2;
|
|
|
|
} else if (timer_id == BX_SER_THIS s[3].tx_timer_index) {
|
|
|
|
port = 3;
|
|
|
|
}
|
2004-12-05 23:23:39 +03:00
|
|
|
|
2004-01-17 18:51:09 +03:00
|
|
|
if (BX_SER_THIS s[port].modem_cntl.local_loopback) {
|
2004-01-18 03:18:44 +03:00
|
|
|
rx_fifo_enq(port, BX_SER_THIS s[port].tsrbuffer);
|
2001-04-10 05:04:59 +04:00
|
|
|
} else {
|
2004-07-28 23:36:42 +04:00
|
|
|
switch (BX_SER_THIS s[port].io_mode) {
|
|
|
|
case BX_SER_MODE_FILE:
|
|
|
|
fputc(BX_SER_THIS s[port].tsrbuffer, BX_SER_THIS s[port].output);
|
|
|
|
fflush(BX_SER_THIS s[port].output);
|
|
|
|
break;
|
|
|
|
case BX_SER_MODE_TERM:
|
|
|
|
#if defined(SERIAL_ENABLE)
|
|
|
|
BX_DEBUG(("com%d: write: '%c'", port+1, BX_SER_THIS s[port].tsrbuffer));
|
|
|
|
if (BX_SER_THIS s[port].tty_id >= 0) {
|
|
|
|
write(BX_SER_THIS s[port].tty_id, (bx_ptr_t) & BX_SER_THIS s[port].tsrbuffer, 1);
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
break;
|
|
|
|
case BX_SER_MODE_RAW:
|
2003-10-12 14:51:58 +04:00
|
|
|
#if USE_RAW_SERIAL
|
2004-07-28 23:36:42 +04:00
|
|
|
if (!BX_SER_THIS s[port].raw->ready_transmit())
|
|
|
|
BX_PANIC(("com%d: not ready to transmit", port+1));
|
|
|
|
BX_SER_THIS s[port].raw->transmit(BX_SER_THIS s[port].tsrbuffer);
|
2001-04-10 05:04:59 +04:00
|
|
|
#endif
|
2004-07-28 23:36:42 +04:00
|
|
|
break;
|
2004-11-27 13:09:41 +03:00
|
|
|
case BX_SER_MODE_MOUSE:
|
|
|
|
BX_INFO(("com%d: write to mouse ignored: 0x%02x", port+1, BX_SER_THIS s[port].tsrbuffer));
|
|
|
|
break;
|
2005-07-10 20:51:09 +04:00
|
|
|
case BX_SER_MODE_SOCKET:
|
|
|
|
if (BX_SER_THIS s[port].socket_id >= 0) {
|
|
|
|
#ifdef WIN32
|
|
|
|
BX_INFO(("attempting to write win32 : %c", BX_SER_THIS s[port].tsrbuffer));
|
|
|
|
::send(BX_SER_THIS s[port].socket_id,
|
|
|
|
(const char*) & BX_SER_THIS s[port].tsrbuffer, 1, 0);
|
|
|
|
#else
|
|
|
|
::write(BX_SER_THIS s[port].socket_id,
|
|
|
|
(bx_ptr_t) & BX_SER_THIS s[port].tsrbuffer, 1);
|
|
|
|
#endif
|
|
|
|
}
|
2008-05-22 12:13:22 +04:00
|
|
|
case BX_SER_MODE_PIPE:
|
|
|
|
#ifdef WIN32
|
|
|
|
if (BX_SER_THIS s[port].pipe) {
|
|
|
|
DWORD written;
|
|
|
|
WriteFile(BX_SER_THIS s[port].pipe, (bx_ptr_t)& BX_SER_THIS s[port].tsrbuffer, 1, &written, NULL);
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
break;
|
2004-07-28 23:36:42 +04:00
|
|
|
}
|
2001-04-10 05:04:59 +04:00
|
|
|
}
|
|
|
|
|
2004-01-17 18:51:09 +03:00
|
|
|
BX_SER_THIS s[port].line_status.tsr_empty = 1;
|
|
|
|
if (BX_SER_THIS s[port].fifo_cntl.enable && (BX_SER_THIS s[port].tx_fifo_end > 0)) {
|
|
|
|
BX_SER_THIS s[port].tsrbuffer = BX_SER_THIS s[port].tx_fifo[0];
|
|
|
|
BX_SER_THIS s[port].line_status.tsr_empty = 0;
|
|
|
|
memcpy(&BX_SER_THIS s[port].tx_fifo[0], &BX_SER_THIS s[port].tx_fifo[1], 15);
|
|
|
|
gen_int = (--BX_SER_THIS s[port].tx_fifo_end == 0);
|
|
|
|
} else if (!BX_SER_THIS s[port].line_status.thr_empty) {
|
|
|
|
BX_SER_THIS s[port].tsrbuffer = BX_SER_THIS s[port].thrbuffer;
|
|
|
|
BX_SER_THIS s[port].line_status.tsr_empty = 0;
|
2003-11-09 03:14:43 +03:00
|
|
|
gen_int = 1;
|
|
|
|
}
|
2004-01-17 18:51:09 +03:00
|
|
|
if (!BX_SER_THIS s[port].line_status.tsr_empty) {
|
2003-11-09 03:14:43 +03:00
|
|
|
if (gen_int) {
|
2004-01-17 18:51:09 +03:00
|
|
|
BX_SER_THIS s[port].line_status.thr_empty = 1;
|
|
|
|
raise_interrupt(port, BX_SER_INT_TXHOLD);
|
2003-11-09 03:14:43 +03:00
|
|
|
}
|
2004-01-17 18:51:09 +03:00
|
|
|
bx_pc_system.activate_timer(BX_SER_THIS s[port].tx_timer_index,
|
|
|
|
(int) (1000000.0 / BX_SER_THIS s[port].baudrate *
|
|
|
|
(BX_SER_THIS s[port].line_cntl.wordlen_sel + 5)),
|
2003-09-15 00:16:25 +04:00
|
|
|
0); /* not continuous */
|
|
|
|
}
|
2001-04-10 05:04:59 +04:00
|
|
|
}
|
|
|
|
|
2011-04-30 22:22:35 +04:00
|
|
|
void bx_serial_c::rx_timer_handler(void *this_ptr)
|
2001-04-10 05:04:59 +04:00
|
|
|
{
|
|
|
|
bx_serial_c *class_ptr = (bx_serial_c *) this_ptr;
|
|
|
|
|
|
|
|
class_ptr->rx_timer();
|
|
|
|
}
|
|
|
|
|
2011-04-30 22:22:35 +04:00
|
|
|
void bx_serial_c::rx_timer(void)
|
2001-04-10 05:04:59 +04:00
|
|
|
{
|
2004-03-13 20:17:16 +03:00
|
|
|
#if BX_HAVE_SELECT && defined(SERIAL_ENABLE)
|
2001-06-10 05:35:10 +04:00
|
|
|
struct timeval tval;
|
2001-04-10 05:04:59 +04:00
|
|
|
fd_set fds;
|
|
|
|
#endif
|
2004-01-17 18:51:09 +03:00
|
|
|
Bit8u port = 0;
|
|
|
|
int timer_id;
|
2004-07-28 23:36:42 +04:00
|
|
|
bx_bool data_ready = 0;
|
2004-01-17 18:51:09 +03:00
|
|
|
|
|
|
|
timer_id = bx_pc_system.triggeredTimerID();
|
|
|
|
if (timer_id == BX_SER_THIS s[0].rx_timer_index) {
|
|
|
|
port = 0;
|
|
|
|
} else if (timer_id == BX_SER_THIS s[1].rx_timer_index) {
|
|
|
|
port = 1;
|
|
|
|
} else if (timer_id == BX_SER_THIS s[2].rx_timer_index) {
|
|
|
|
port = 2;
|
|
|
|
} else if (timer_id == BX_SER_THIS s[3].rx_timer_index) {
|
|
|
|
port = 3;
|
|
|
|
}
|
2004-12-05 23:23:39 +03:00
|
|
|
|
2004-01-17 18:51:09 +03:00
|
|
|
int bdrate = BX_SER_THIS s[port].baudrate / (BX_SER_THIS s[port].line_cntl.wordlen_sel + 5);
|
2001-05-23 11:48:11 +04:00
|
|
|
unsigned char chbuf = 0;
|
2001-04-10 05:04:59 +04:00
|
|
|
|
2004-07-28 23:36:42 +04:00
|
|
|
if (BX_SER_THIS s[port].io_mode == BX_SER_MODE_TERM) {
|
2004-03-13 20:17:16 +03:00
|
|
|
#if BX_HAVE_SELECT && defined(SERIAL_ENABLE)
|
2004-07-28 23:36:42 +04:00
|
|
|
tval.tv_sec = 0;
|
|
|
|
tval.tv_usec = 0;
|
2001-04-10 05:04:59 +04:00
|
|
|
|
|
|
|
// MacOS: I'm not sure what to do with this, since I don't know
|
|
|
|
// what an fd_set is or what FD_SET() or select() do. They aren't
|
|
|
|
// declared in the CodeWarrior standard library headers. I'm just
|
|
|
|
// leaving it commented out for the moment.
|
|
|
|
|
2004-07-28 23:36:42 +04:00
|
|
|
FD_ZERO(&fds);
|
|
|
|
if (BX_SER_THIS s[port].tty_id >= 0) FD_SET(BX_SER_THIS s[port].tty_id, &fds);
|
2004-03-13 20:17:16 +03:00
|
|
|
#endif
|
2004-07-28 23:36:42 +04:00
|
|
|
}
|
2004-01-17 18:51:09 +03:00
|
|
|
if ((BX_SER_THIS s[port].line_status.rxdata_ready == 0) ||
|
|
|
|
(BX_SER_THIS s[port].fifo_cntl.enable)) {
|
2004-11-27 13:09:41 +03:00
|
|
|
switch (BX_SER_THIS s[port].io_mode) {
|
2005-07-10 20:51:09 +04:00
|
|
|
case BX_SER_MODE_SOCKET:
|
2008-08-22 11:58:20 +04:00
|
|
|
#if BX_HAVE_SELECT && defined(SERIAL_ENABLE)
|
2005-07-10 20:51:09 +04:00
|
|
|
if (BX_SER_THIS s[port].line_status.rxdata_ready == 0) {
|
|
|
|
tval.tv_sec = 0;
|
|
|
|
tval.tv_usec = 0;
|
|
|
|
FD_ZERO(&fds);
|
2010-11-23 17:59:36 +03:00
|
|
|
SOCKET socketid = BX_SER_THIS s[port].socket_id;
|
2008-01-27 01:24:03 +03:00
|
|
|
if (socketid >= 0) FD_SET(socketid, &fds);
|
2005-07-10 20:51:09 +04:00
|
|
|
if ((socketid >= 0) && (select(socketid+1, &fds, NULL, NULL, &tval) == 1)) {
|
2011-04-30 22:22:35 +04:00
|
|
|
ssize_t bytes = (ssize_t)
|
2008-01-27 01:24:03 +03:00
|
|
|
#ifdef WIN32
|
2011-04-30 22:22:35 +04:00
|
|
|
::recv(socketid, (char*) &chbuf, 1, 0);
|
2005-07-10 20:51:09 +04:00
|
|
|
#else
|
2011-04-30 22:22:35 +04:00
|
|
|
read(socketid, &chbuf, 1);
|
2005-07-10 20:51:09 +04:00
|
|
|
#endif
|
2011-04-30 22:47:04 +04:00
|
|
|
if (bytes > 0) {
|
2011-04-30 22:22:35 +04:00
|
|
|
BX_INFO((" -- COM %d : read byte [%d]", port+1, chbuf));
|
|
|
|
data_ready = 1;
|
|
|
|
}
|
2005-07-10 20:51:09 +04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
#endif
|
2008-01-27 01:24:03 +03:00
|
|
|
break;
|
2004-11-27 13:09:41 +03:00
|
|
|
case BX_SER_MODE_RAW:
|
2003-10-12 14:51:58 +04:00
|
|
|
#if USE_RAW_SERIAL
|
2004-11-27 13:09:41 +03:00
|
|
|
int data;
|
|
|
|
if ((data_ready = BX_SER_THIS s[port].raw->ready_receive())) {
|
|
|
|
data = BX_SER_THIS s[port].raw->receive();
|
2008-02-16 01:05:43 +03:00
|
|
|
if (data < 0) {
|
2004-11-27 13:09:41 +03:00
|
|
|
data_ready = 0;
|
|
|
|
switch (data) {
|
|
|
|
case RAW_EVENT_BREAK:
|
|
|
|
BX_SER_THIS s[port].line_status.break_int = 1;
|
|
|
|
raise_interrupt(port, BX_SER_INT_RXLSTAT);
|
|
|
|
break;
|
|
|
|
case RAW_EVENT_FRAME:
|
|
|
|
BX_SER_THIS s[port].line_status.framing_error = 1;
|
|
|
|
raise_interrupt(port, BX_SER_INT_RXLSTAT);
|
|
|
|
break;
|
|
|
|
case RAW_EVENT_OVERRUN:
|
|
|
|
BX_SER_THIS s[port].line_status.overrun_error = 1;
|
|
|
|
raise_interrupt(port, BX_SER_INT_RXLSTAT);
|
|
|
|
break;
|
|
|
|
case RAW_EVENT_PARITY:
|
|
|
|
BX_SER_THIS s[port].line_status.parity_error = 1;
|
|
|
|
raise_interrupt(port, BX_SER_INT_RXLSTAT);
|
|
|
|
break;
|
|
|
|
case RAW_EVENT_CTS_ON:
|
|
|
|
case RAW_EVENT_CTS_OFF:
|
|
|
|
case RAW_EVENT_DSR_ON:
|
|
|
|
case RAW_EVENT_DSR_OFF:
|
|
|
|
case RAW_EVENT_RING_ON:
|
|
|
|
case RAW_EVENT_RING_OFF:
|
|
|
|
case RAW_EVENT_RLSD_ON:
|
|
|
|
case RAW_EVENT_RLSD_OFF:
|
|
|
|
raise_interrupt(port, BX_SER_INT_MODSTAT);
|
|
|
|
break;
|
|
|
|
}
|
2004-07-28 23:36:42 +04:00
|
|
|
}
|
2004-03-28 16:41:12 +04:00
|
|
|
}
|
2004-11-27 13:09:41 +03:00
|
|
|
if (data_ready) {
|
|
|
|
chbuf = data;
|
|
|
|
}
|
2001-04-10 05:04:59 +04:00
|
|
|
#endif
|
2004-11-27 13:09:41 +03:00
|
|
|
break;
|
|
|
|
case BX_SER_MODE_TERM:
|
2004-07-28 23:36:42 +04:00
|
|
|
#if BX_HAVE_SELECT && defined(SERIAL_ENABLE)
|
2004-11-27 13:09:41 +03:00
|
|
|
if ((BX_SER_THIS s[port].tty_id >= 0) && (select(BX_SER_THIS s[port].tty_id + 1, &fds, NULL, NULL, &tval) == 1)) {
|
|
|
|
(void) read(BX_SER_THIS s[port].tty_id, &chbuf, 1);
|
|
|
|
BX_DEBUG(("com%d: read: '%c'", port+1, chbuf));
|
|
|
|
data_ready = 1;
|
|
|
|
}
|
2004-07-28 23:36:42 +04:00
|
|
|
#endif
|
2004-11-27 13:09:41 +03:00
|
|
|
break;
|
|
|
|
case BX_SER_MODE_MOUSE:
|
2004-12-03 00:34:26 +03:00
|
|
|
if (BX_SER_THIS mouse_internal_buffer.num_elements > 0) {
|
|
|
|
chbuf = BX_SER_THIS mouse_internal_buffer.buffer[BX_SER_THIS mouse_internal_buffer.head];
|
|
|
|
BX_SER_THIS mouse_internal_buffer.head = (BX_SER_THIS mouse_internal_buffer.head + 1) %
|
|
|
|
BX_MOUSE_BUFF_SIZE;
|
|
|
|
BX_SER_THIS mouse_internal_buffer.num_elements--;
|
|
|
|
data_ready = 1;
|
|
|
|
}
|
2004-11-27 13:09:41 +03:00
|
|
|
break;
|
2008-05-22 12:13:22 +04:00
|
|
|
case BX_SER_MODE_PIPE:
|
|
|
|
#ifdef WIN32
|
|
|
|
DWORD avail = 0;
|
|
|
|
if (BX_SER_THIS s[port].pipe &&
|
|
|
|
PeekNamedPipe(BX_SER_THIS s[port].pipe, NULL, 0, NULL, &avail, NULL) &&
|
|
|
|
avail > 0) {
|
|
|
|
ReadFile(BX_SER_THIS s[port].pipe, &chbuf, 1, &avail, NULL);
|
|
|
|
data_ready = 1;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
break;
|
2004-07-28 23:36:42 +04:00
|
|
|
}
|
|
|
|
if (data_ready) {
|
2004-01-17 18:51:09 +03:00
|
|
|
if (!BX_SER_THIS s[port].modem_cntl.local_loopback) {
|
2004-01-18 03:18:44 +03:00
|
|
|
rx_fifo_enq(port, chbuf);
|
2001-04-10 05:04:59 +04:00
|
|
|
}
|
|
|
|
} else {
|
2004-01-17 18:51:09 +03:00
|
|
|
if (!BX_SER_THIS s[port].fifo_cntl.enable) {
|
2003-11-09 03:14:43 +03:00
|
|
|
bdrate = (int) (1000000.0 / 100000); // Poll frequency is 100ms
|
|
|
|
}
|
2001-04-10 05:04:59 +04:00
|
|
|
}
|
|
|
|
} else {
|
|
|
|
// Poll at 4x baud rate to see if the next-char can
|
|
|
|
// be read
|
2002-01-20 19:35:32 +03:00
|
|
|
bdrate *= 4;
|
2001-04-10 05:04:59 +04:00
|
|
|
}
|
|
|
|
|
2004-01-17 18:51:09 +03:00
|
|
|
bx_pc_system.activate_timer(BX_SER_THIS s[port].rx_timer_index,
|
2011-04-30 22:22:35 +04:00
|
|
|
(int) (1000000.0 / bdrate), 0); /* not continuous */
|
2001-04-10 05:04:59 +04:00
|
|
|
}
|
2003-11-09 03:14:43 +03:00
|
|
|
|
2011-04-30 22:22:35 +04:00
|
|
|
void bx_serial_c::fifo_timer_handler(void *this_ptr)
|
2003-11-09 03:14:43 +03:00
|
|
|
{
|
|
|
|
bx_serial_c *class_ptr = (bx_serial_c *) this_ptr;
|
|
|
|
|
|
|
|
class_ptr->fifo_timer();
|
|
|
|
}
|
|
|
|
|
2011-04-30 22:22:35 +04:00
|
|
|
void bx_serial_c::fifo_timer(void)
|
2003-11-09 03:14:43 +03:00
|
|
|
{
|
2004-01-17 18:51:09 +03:00
|
|
|
Bit8u port = 0;
|
|
|
|
int timer_id;
|
|
|
|
|
|
|
|
timer_id = bx_pc_system.triggeredTimerID();
|
|
|
|
if (timer_id == BX_SER_THIS s[0].fifo_timer_index) {
|
|
|
|
port = 0;
|
|
|
|
} else if (timer_id == BX_SER_THIS s[1].fifo_timer_index) {
|
|
|
|
port = 1;
|
|
|
|
} else if (timer_id == BX_SER_THIS s[2].fifo_timer_index) {
|
|
|
|
port = 2;
|
|
|
|
} else if (timer_id == BX_SER_THIS s[3].fifo_timer_index) {
|
|
|
|
port = 3;
|
|
|
|
}
|
|
|
|
BX_SER_THIS s[port].line_status.rxdata_ready = 1;
|
|
|
|
raise_interrupt(port, BX_SER_INT_FIFO);
|
2003-11-09 03:14:43 +03:00
|
|
|
}
|
2004-12-03 00:34:26 +03:00
|
|
|
|
2012-06-21 21:33:37 +04:00
|
|
|
void bx_serial_c::mouse_enq_static(void *dev, int delta_x, int delta_y, int delta_z, unsigned button_state, bx_bool absxy)
|
2009-03-03 23:34:50 +03:00
|
|
|
{
|
2012-06-21 21:33:37 +04:00
|
|
|
((bx_serial_c*)dev)->mouse_enq(delta_x, delta_y, delta_z, button_state, absxy);
|
2009-03-03 23:34:50 +03:00
|
|
|
}
|
|
|
|
|
2012-06-21 21:33:37 +04:00
|
|
|
void bx_serial_c::mouse_enq(int delta_x, int delta_y, int delta_z, unsigned button_state, bx_bool absxy)
|
2004-12-03 00:34:26 +03:00
|
|
|
{
|
2008-06-18 20:33:44 +04:00
|
|
|
Bit8u b1, b2, b3, mouse_data[5];
|
2006-12-31 14:56:14 +03:00
|
|
|
int bytes, tail;
|
2004-12-03 00:34:26 +03:00
|
|
|
|
|
|
|
if (BX_SER_THIS mouse_port == -1) {
|
|
|
|
BX_ERROR(("mouse not connected to a serial port"));
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2004-12-08 00:06:35 +03:00
|
|
|
// if the DTR and RTS lines aren't up, the mouse doesn't have any power to send packets.
|
|
|
|
if (!BX_SER_THIS s[BX_SER_THIS mouse_port].modem_cntl.dtr || !BX_SER_THIS s[BX_SER_THIS mouse_port].modem_cntl.rts)
|
|
|
|
return;
|
|
|
|
|
2004-12-03 00:34:26 +03:00
|
|
|
// scale down the motion
|
2008-02-16 01:05:43 +03:00
|
|
|
if ((delta_x < -1) || (delta_x > 1))
|
2004-12-03 00:34:26 +03:00
|
|
|
delta_x /= 2;
|
2008-02-16 01:05:43 +03:00
|
|
|
if ((delta_y < -1) || (delta_y > 1))
|
2004-12-03 00:34:26 +03:00
|
|
|
delta_y /= 2;
|
|
|
|
|
|
|
|
if(delta_x>127) delta_x=127;
|
|
|
|
if(delta_y>127) delta_y=127;
|
|
|
|
if(delta_x<-128) delta_x=-128;
|
|
|
|
if(delta_y<-128) delta_y=-128;
|
|
|
|
|
|
|
|
BX_SER_THIS mouse_delayed_dx+=delta_x;
|
|
|
|
BX_SER_THIS mouse_delayed_dy-=delta_y;
|
2004-12-08 00:06:35 +03:00
|
|
|
BX_SER_THIS mouse_delayed_dz =delta_z;
|
2004-12-03 00:34:26 +03:00
|
|
|
|
2004-12-08 00:06:35 +03:00
|
|
|
if ((BX_SER_THIS mouse_internal_buffer.num_elements + 4) >= BX_MOUSE_BUFF_SIZE) {
|
2004-12-03 00:34:26 +03:00
|
|
|
return; /* buffer doesn't have the space */
|
|
|
|
}
|
|
|
|
|
|
|
|
if (BX_SER_THIS mouse_delayed_dx > 127) {
|
|
|
|
delta_x = 127;
|
|
|
|
BX_SER_THIS mouse_delayed_dx -= 127;
|
|
|
|
} else if (BX_SER_THIS mouse_delayed_dx < -128) {
|
|
|
|
delta_x = -128;
|
|
|
|
BX_SER_THIS mouse_delayed_dx += 128;
|
|
|
|
} else {
|
|
|
|
delta_x = BX_SER_THIS mouse_delayed_dx;
|
|
|
|
BX_SER_THIS mouse_delayed_dx = 0;
|
|
|
|
}
|
|
|
|
if (BX_SER_THIS mouse_delayed_dy > 127) {
|
|
|
|
delta_y = 127;
|
|
|
|
BX_SER_THIS mouse_delayed_dy -= 127;
|
|
|
|
} else if (BX_SER_THIS mouse_delayed_dy < -128) {
|
|
|
|
delta_y = -128;
|
|
|
|
BX_SER_THIS mouse_delayed_dy += 128;
|
|
|
|
} else {
|
|
|
|
delta_y = BX_SER_THIS mouse_delayed_dy;
|
|
|
|
BX_SER_THIS mouse_delayed_dy = 0;
|
|
|
|
}
|
|
|
|
|
2006-12-31 14:56:14 +03:00
|
|
|
if (BX_SER_THIS mouse_type != BX_MOUSE_TYPE_SERIAL_MSYS) {
|
|
|
|
b1 = (Bit8u) delta_x;
|
|
|
|
b2 = (Bit8u) delta_y;
|
|
|
|
b3 = (Bit8u) -((Bit8s) delta_z);
|
|
|
|
mouse_data[0] = 0x40 | ((b1 & 0xc0) >> 6) | ((b2 & 0xc0) >> 4);
|
|
|
|
mouse_data[0] |= ((button_state & 0x01) << 5) | ((button_state & 0x02) << 3);
|
|
|
|
mouse_data[1] = b1 & 0x3f;
|
|
|
|
mouse_data[2] = b2 & 0x3f;
|
|
|
|
mouse_data[3] = b3 & 0x0f;
|
|
|
|
mouse_data[3] |= ((button_state & 0x04) << 2);
|
|
|
|
bytes = 3;
|
|
|
|
if (BX_SER_THIS mouse_type == BX_MOUSE_TYPE_SERIAL_WHEEL) bytes = 4;
|
|
|
|
} else {
|
|
|
|
b1 = (Bit8u) (delta_x / 2);
|
|
|
|
b2 = (Bit8u) -((Bit8s) (delta_y / 2));
|
|
|
|
mouse_data[0] = 0x80 | ((~button_state & 0x01) << 2);
|
|
|
|
mouse_data[0] |= ((~button_state & 0x06) >> 1);
|
|
|
|
mouse_data[1] = b1;
|
|
|
|
mouse_data[2] = b2;
|
|
|
|
mouse_data[3] = 0;
|
|
|
|
mouse_data[4] = 0;
|
|
|
|
bytes = 5;
|
|
|
|
}
|
2004-12-03 00:34:26 +03:00
|
|
|
|
|
|
|
/* enqueue mouse data in multibyte internal mouse buffer */
|
2004-12-05 23:23:39 +03:00
|
|
|
for (int i = 0; i < bytes; i++) {
|
2004-12-03 00:34:26 +03:00
|
|
|
tail = (BX_SER_THIS mouse_internal_buffer.head + BX_SER_THIS mouse_internal_buffer.num_elements) %
|
|
|
|
BX_MOUSE_BUFF_SIZE;
|
|
|
|
BX_SER_THIS mouse_internal_buffer.buffer[tail] = mouse_data[i];
|
|
|
|
BX_SER_THIS mouse_internal_buffer.num_elements++;
|
|
|
|
}
|
|
|
|
}
|