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
|
|
|
/////////////////////////////////////////////////////////////////////////
|
|
|
|
//
|
2015-10-06 21:20:16 +03:00
|
|
|
// Copyright (C) 2001-2015 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
|
|
|
|
|
2014-06-08 12:40:08 +04:00
|
|
|
#include "iodev.h"
|
|
|
|
|
2013-11-01 22:19:52 +04:00
|
|
|
#if !defined(WIN32) || defined(__CYGWIN__)
|
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;
|
2013-11-01 20:26:59 +04:00
|
|
|
#else
|
2014-02-15 16:18:15 +04:00
|
|
|
#include <winsock2.h>
|
2013-11-01 22:19:52 +04:00
|
|
|
#define BX_SER_WIN32
|
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 "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",
|
|
|
|
"term",
|
|
|
|
"raw",
|
|
|
|
"mouse",
|
|
|
|
"socket-client",
|
|
|
|
"socket-server",
|
2013-02-23 19:15:59 +04:00
|
|
|
"pipe-client",
|
|
|
|
"pipe-server",
|
2012-04-16 23:17:10 +04:00
|
|
|
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,
|
2013-02-23 19:15:59 +04:00
|
|
|
"The mode can be one these: 'null', 'file', 'term', 'raw', 'mouse', 'socket*', 'pipe*'",
|
|
|
|
serial_mode_list, BX_SER_MODE_NULL, BX_SER_MODE_NULL);
|
2012-04-16 23:17:10 +04:00
|
|
|
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);
|
|
|
|
enabled->set_dependent_list(deplist);
|
2013-12-01 22:26:37 +04:00
|
|
|
deplist = new bx_list_c(NULL);
|
|
|
|
deplist->add(path);
|
|
|
|
mode->set_dependent_list(deplist, 1);
|
|
|
|
mode->set_dependent_bitmap(BX_SER_MODE_NULL, 0);
|
|
|
|
mode->set_dependent_bitmap(BX_SER_MODE_MOUSE, 0);
|
2012-04-16 23:17:10 +04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
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++) {
|
2013-01-26 22:17:23 +04:00
|
|
|
if (SIM->parse_param_from_list(context, params[i], base) < 0) {
|
2012-04-16 23:17:10 +04:00
|
|
|
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)
|
|
|
|
{
|
2013-12-01 22:26:37 +04:00
|
|
|
char port[20];
|
2012-04-16 23:17:10 +04:00
|
|
|
|
|
|
|
for (int i=0; i<BX_N_SERIAL_PORTS; i++) {
|
2013-12-01 22:26:37 +04:00
|
|
|
sprintf(port, "ports.serial.%d", i+1);
|
|
|
|
bx_list_c *base = (bx_list_c*) SIM->get_param(port);
|
|
|
|
sprintf(port, "com%d", i+1);
|
|
|
|
SIM->write_param_list(fp, base, port, 0);
|
2012-04-16 23:17:10 +04:00
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
// device plugin entry points
|
|
|
|
|
2014-06-08 12:40:08 +04:00
|
|
|
int CDECL 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);
|
2013-12-01 22:26:37 +04:00
|
|
|
return 0; // Success
|
2002-10-25 01:07:56 +04:00
|
|
|
}
|
|
|
|
|
2014-06-08 12:40:08 +04:00
|
|
|
void CDECL libserial_LTX_plugin_fini(void)
|
2002-10-25 01:07:56 +04:00
|
|
|
{
|
2013-12-01 22:26:37 +04:00
|
|
|
char port[6];
|
2012-04-16 23:17:10 +04:00
|
|
|
|
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++) {
|
2013-12-01 22:26:37 +04:00
|
|
|
sprintf(port, "com%d", i+1);
|
|
|
|
SIM->unregister_addon_option(port);
|
|
|
|
sprintf(port, "%d", i+1);
|
|
|
|
menu->remove(port);
|
2012-04-16 23:17:10 +04:00
|
|
|
}
|
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:
|
2013-11-01 22:19:52 +04:00
|
|
|
#if defined(SERIAL_ENABLE) && !defined(BX_SER_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;
|
2013-02-23 19:15:59 +04:00
|
|
|
case BX_SER_MODE_SOCKET_CLIENT:
|
|
|
|
case BX_SER_MODE_SOCKET_SERVER:
|
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;
|
2013-02-23 19:15:59 +04:00
|
|
|
case BX_SER_MODE_PIPE_CLIENT:
|
|
|
|
case BX_SER_MODE_PIPE_SERVER:
|
2013-11-01 22:19:52 +04:00
|
|
|
#ifdef BX_SER_WIN32
|
2008-05-22 12:13:22 +04:00
|
|
|
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
|
|
|
}
|
2012-08-19 12:16:20 +04:00
|
|
|
SIM->get_bochs_root()->remove("serial");
|
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;
|
2014-01-26 17:48:10 +04:00
|
|
|
BX_SER_THIS mouse_delayed_dz = 0;
|
|
|
|
BX_SER_THIS mouse_buttons = 0;
|
|
|
|
BX_SER_THIS mouse_update = 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
|
2014-01-17 22:25:13 +04:00
|
|
|
bx_pc_system.setTimerParam(BX_SER_THIS s[i].tx_timer_index, i);
|
2003-10-30 00:00:04 +03:00
|
|
|
}
|
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
|
2014-01-17 22:25:13 +04:00
|
|
|
bx_pc_system.setTimerParam(BX_SER_THIS s[i].rx_timer_index, i);
|
2003-10-30 00:00:04 +03:00
|
|
|
}
|
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
|
2014-01-17 22:25:13 +04:00
|
|
|
bx_pc_system.setTimerParam(BX_SER_THIS s[i].fifo_timer_index, i);
|
2003-11-09 03:14:43 +03:00
|
|
|
}
|
2003-10-30 00:00:04 +03:00
|
|
|
|
|
|
|
/* 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;
|
2014-01-26 01:05:49 +04:00
|
|
|
BX_SER_THIS s[i].databyte_usec = 87;
|
2003-10-31 20:23:56 +03:00
|
|
|
|
2013-06-16 12:36:47 +04:00
|
|
|
for (unsigned addr = ports[i]; addr < (unsigned)(ports[i] + 8); addr++) {
|
|
|
|
BX_DEBUG(("com%d initialize register for read/write: 0x%04x", i + 1, addr));
|
2013-06-25 21:32:33 +04:00
|
|
|
if (addr < (unsigned)(ports[i] + 7)) {
|
2013-06-16 12:36:47 +04:00
|
|
|
DEV_register_ioread_handler(this, read_handler, addr, name, 3);
|
|
|
|
DEV_register_iowrite_handler(this, write_handler, addr, name, 3);
|
|
|
|
} else {
|
|
|
|
DEV_register_ioread_handler(this, read_handler, addr, name, 1);
|
|
|
|
DEV_register_iowrite_handler(this, write_handler, addr, name, 1);
|
|
|
|
}
|
2003-10-30 00:00:04 +03:00
|
|
|
}
|
2004-07-28 23:36:42 +04:00
|
|
|
|
|
|
|
BX_SER_THIS s[i].io_mode = BX_SER_MODE_NULL;
|
2013-02-23 19:15:59 +04:00
|
|
|
Bit8u mode = SIM->get_param_enum("mode", base)->get();
|
2013-12-01 22:26:37 +04:00
|
|
|
bx_param_string_c *devparam = SIM->get_param_string("dev", base);
|
|
|
|
const char *dev = devparam->getptr();
|
2013-02-23 19:15:59 +04:00
|
|
|
if (mode == BX_SER_MODE_FILE) {
|
2013-12-01 22:26:37 +04:00
|
|
|
if (!devparam->isempty()) {
|
|
|
|
// tx_timer() opens the output file on demand
|
|
|
|
BX_SER_THIS s[i].io_mode = BX_SER_MODE_FILE;
|
2004-07-28 23:36:42 +04:00
|
|
|
}
|
2013-02-23 19:15:59 +04:00
|
|
|
} else if (mode == BX_SER_MODE_TERM) {
|
2013-11-01 22:19:52 +04:00
|
|
|
#if defined(SERIAL_ENABLE) && !defined(BX_SER_WIN32)
|
2013-12-01 22:26:37 +04:00
|
|
|
if (!devparam->isempty()) {
|
2006-03-01 20:14:36 +03:00
|
|
|
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 */
|
2013-02-23 19:15:59 +04:00
|
|
|
} else if (mode == BX_SER_MODE_RAW) {
|
2004-07-28 23:36:42 +04:00
|
|
|
#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
|
2013-02-23 19:15:59 +04:00
|
|
|
} else if (mode == BX_SER_MODE_MOUSE) {
|
2004-11-27 13:09:41 +03:00
|
|
|
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();
|
2013-02-23 19:15:59 +04:00
|
|
|
} else if ((mode == BX_SER_MODE_SOCKET_CLIENT) ||
|
|
|
|
(mode == BX_SER_MODE_SOCKET_SERVER)) {
|
|
|
|
BX_SER_THIS s[i].io_mode = mode;
|
2005-07-10 20:51:09 +04:00
|
|
|
struct sockaddr_in sin;
|
|
|
|
struct hostent *hp;
|
|
|
|
char host[BX_PATHNAME_LEN];
|
|
|
|
int port;
|
2010-11-23 17:59:36 +03:00
|
|
|
SOCKET socket;
|
2013-02-23 19:15:59 +04:00
|
|
|
bx_bool server = (mode == BX_SER_MODE_SOCKET_SERVER);
|
2005-07-10 20:51:09 +04:00
|
|
|
|
2013-11-01 22:19:52 +04:00
|
|
|
#ifdef BX_SER_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));
|
2013-02-23 19:15:59 +04:00
|
|
|
} else if ((mode == BX_SER_MODE_PIPE_CLIENT) ||
|
|
|
|
(mode == BX_SER_MODE_PIPE_SERVER)) {
|
2008-05-22 12:13:22 +04:00
|
|
|
if (strlen(dev) > 0) {
|
2013-02-23 19:15:59 +04:00
|
|
|
bx_bool server = (mode == BX_SER_MODE_PIPE_SERVER);
|
2013-11-01 22:19:52 +04:00
|
|
|
#ifdef BX_SER_WIN32
|
2008-05-22 12:13:22 +04:00
|
|
|
HANDLE pipe;
|
|
|
|
|
2013-02-23 19:15:59 +04:00
|
|
|
BX_SER_THIS s[i].io_mode = mode;
|
2008-05-22 12:13:22 +04:00
|
|
|
|
|
|
|
// 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
|
2013-02-23 19:15:59 +04:00
|
|
|
BX_PANIC(("support for serial mode 'pipe-%s' not available", server?"server":"client"));
|
2008-05-22 12:13:22 +04:00
|
|
|
#endif
|
|
|
|
}
|
2013-02-23 19:15:59 +04:00
|
|
|
} else if (mode != BX_SER_MODE_NULL) {
|
|
|
|
BX_PANIC(("unknown serial i/o mode %d", 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++;
|
2013-06-16 12:36:47 +04:00
|
|
|
BX_INFO(("com%d at 0x%04x irq %d (mode: %s)", i+1, ports[i], BX_SER_THIS s[i].IRQ,
|
|
|
|
SIM->get_param_enum("mode", base)->get_selected()));
|
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)
|
|
|
|
{
|
2015-10-06 21:20:16 +03:00
|
|
|
unsigned i;
|
2006-05-27 19:54:49 +04:00
|
|
|
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++) {
|
2015-01-26 00:24:13 +03:00
|
|
|
sprintf(name, "%u", 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);
|
2014-01-26 01:05:49 +04:00
|
|
|
new bx_shadow_num_c(port, "databyte_usec", &BX_SER_THIS s[i].databyte_usec);
|
2006-05-27 19:54:49 +04:00
|
|
|
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);
|
2015-10-06 21:20:16 +03:00
|
|
|
new bx_shadow_data_c(port, "rx_fifo", BX_SER_THIS s[i].rx_fifo, 16, 1);
|
|
|
|
new bx_shadow_data_c(port, "tx_fifo", BX_SER_THIS s[i].tx_fifo, 16, 1);
|
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);
|
2014-01-26 17:48:10 +04:00
|
|
|
new bx_shadow_num_c(list, "mouse_buttons", &BX_SER_THIS mouse_buttons);
|
|
|
|
new bx_shadow_bool_c(list, "mouse_update", &BX_SER_THIS mouse_update);
|
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);
|
2015-10-06 21:20:16 +03:00
|
|
|
new bx_shadow_data_c(mousebuf, "buffer", BX_SER_THIS mouse_internal_buffer.buffer,
|
|
|
|
BX_MOUSE_BUFF_SIZE, 1);
|
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;
|
2013-06-16 12:36:47 +04:00
|
|
|
Bit16u ret16;
|
|
|
|
|
|
|
|
if (io_len == 2) {
|
|
|
|
ret16 = BX_SER_THIS read_handler(theSerialDevice, address, 1);
|
|
|
|
ret16 |= (BX_SER_THIS read_handler(theSerialDevice, address + 1, 1)) << 8;
|
|
|
|
return ret16;
|
|
|
|
}
|
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) {
|
2014-12-16 05:26:00 +03:00
|
|
|
memmove(&BX_SER_THIS s[port].rx_fifo[0], &BX_SER_THIS s[port].rx_fifo[1], 15);
|
2004-01-17 18:51:09 +03:00
|
|
|
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;
|
2014-01-26 01:05:49 +04:00
|
|
|
int new_baudrate;
|
|
|
|
bx_bool restart_timer = 0;
|
2002-01-20 19:35:32 +03:00
|
|
|
|
2013-06-16 12:36:47 +04:00
|
|
|
if (io_len == 2) {
|
|
|
|
BX_SER_THIS write_handler(theSerialDevice, address, (value & 0xff), 1);
|
|
|
|
BX_SER_THIS write_handler(theSerialDevice, address + 1, ((value >> 8) & 0xff), 1);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
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
|
|
|
} else {
|
2004-01-17 18:51:09 +03:00
|
|
|
Bit8u bitmask = 0xff >> (3 - BX_SER_THIS s[port].line_cntl.wordlen_sel);
|
2014-01-26 01:05:49 +04:00
|
|
|
value &= bitmask;
|
2004-01-17 18:51:09 +03:00
|
|
|
if (BX_SER_THIS s[port].line_status.thr_empty) {
|
2016-11-23 19:42:22 +03:00
|
|
|
if (BX_SER_THIS s[port].fifo_cntl.enable &&
|
|
|
|
!BX_SER_THIS s[port].modem_cntl.local_loopback) {
|
2014-01-26 01:05:49 +04:00
|
|
|
BX_SER_THIS s[port].tx_fifo[BX_SER_THIS s[port].tx_fifo_end++] = value;
|
2003-11-09 03:14:43 +03:00
|
|
|
} else {
|
2014-01-26 01:05:49 +04:00
|
|
|
BX_SER_THIS s[port].thrbuffer = value;
|
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) {
|
2016-11-23 19:42:22 +03:00
|
|
|
if (BX_SER_THIS s[port].fifo_cntl.enable &&
|
|
|
|
!BX_SER_THIS s[port].modem_cntl.local_loopback) {
|
2004-01-17 18:51:09 +03:00
|
|
|
BX_SER_THIS s[port].tsrbuffer = BX_SER_THIS s[port].tx_fifo[0];
|
2014-12-16 05:26:00 +03:00
|
|
|
memmove(&BX_SER_THIS s[port].tx_fifo[0], &BX_SER_THIS s[port].tx_fifo[1], 15);
|
2004-01-17 18:51:09 +03:00
|
|
|
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
|
|
|
}
|
2016-11-23 19:42:22 +03:00
|
|
|
if (BX_SER_THIS s[port].line_status.thr_empty) {
|
|
|
|
raise_interrupt(port, BX_SER_INT_TXHOLD);
|
|
|
|
}
|
2004-01-17 18:51:09 +03:00
|
|
|
BX_SER_THIS s[port].line_status.tsr_empty = 0;
|
2016-11-23 19:42:22 +03:00
|
|
|
if (BX_SER_THIS s[port].modem_cntl.local_loopback) {
|
|
|
|
rx_fifo_enq(port, BX_SER_THIS s[port].tsrbuffer);
|
|
|
|
BX_SER_THIS s[port].line_status.tsr_empty = 1;
|
|
|
|
} else {
|
|
|
|
bx_pc_system.activate_timer(BX_SER_THIS s[port].tx_timer_index,
|
|
|
|
BX_SER_THIS s[port].databyte_usec,
|
|
|
|
0); /* not continuous */
|
|
|
|
}
|
2003-09-15 00:16:25 +04:00
|
|
|
} 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) {
|
2014-01-26 01:05:49 +04:00
|
|
|
BX_SER_THIS s[port].tx_fifo[BX_SER_THIS s[port].tx_fifo_end++] = value;
|
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
|
|
|
} 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
|
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);
|
|
|
|
}
|
|
|
|
if (!new_b7 && BX_SER_THIS s[port].line_cntl.dlab) {
|
2014-01-26 01:05:49 +04:00
|
|
|
if ((BX_SER_THIS s[port].divisor_lsb | BX_SER_THIS s[port].divisor_msb) != 0) {
|
|
|
|
new_baudrate = (int)(BX_PC_CLOCK_XTL /
|
|
|
|
(16 * ((BX_SER_THIS s[port].divisor_msb << 8) |
|
|
|
|
BX_SER_THIS s[port].divisor_lsb)));
|
|
|
|
if (new_baudrate != BX_SER_THIS s[port].baudrate) {
|
|
|
|
BX_SER_THIS s[port].baudrate = new_baudrate;
|
|
|
|
restart_timer = 1;
|
|
|
|
BX_DEBUG(("com%d: baud rate set to %d", port+1, BX_SER_THIS s[port].baudrate));
|
2004-02-28 16:10:57 +03:00
|
|
|
#if USE_RAW_SERIAL
|
2014-01-26 01:05:49 +04:00
|
|
|
if (BX_SER_THIS s[port].io_mode == BX_SER_MODE_RAW) {
|
|
|
|
BX_SER_THIS s[port].raw->set_baudrate(BX_SER_THIS s[port].baudrate);
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
BX_ERROR(("com%d: ignoring invalid baud rate divisor", port+1));
|
2004-07-28 23:36:42 +04:00
|
|
|
}
|
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;
|
2014-01-26 01:05:49 +04:00
|
|
|
if (new_wordlen != BX_SER_THIS s[port].line_cntl.wordlen_sel) {
|
|
|
|
BX_SER_THIS s[port].line_cntl.wordlen_sel = new_wordlen;
|
|
|
|
restart_timer = 1;
|
|
|
|
}
|
|
|
|
if (restart_timer) {
|
|
|
|
// Start the receive polling process if not already started
|
|
|
|
// and there is a valid baudrate.
|
|
|
|
BX_SER_THIS s[port].databyte_usec = (Bit32u)(1000000.0 / BX_SER_THIS s[port].baudrate *
|
|
|
|
(BX_SER_THIS s[port].line_cntl.wordlen_sel + 7));
|
|
|
|
bx_pc_system.activate_timer(BX_SER_THIS s[port].rx_timer_index,
|
|
|
|
BX_SER_THIS s[port].databyte_usec,
|
|
|
|
0); /* not continuous */
|
|
|
|
}
|
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))) {
|
2013-06-16 12:36:47 +04:00
|
|
|
if (!BX_SER_THIS s[port].modem_cntl.dtr && new_b0) {
|
|
|
|
BX_SER_THIS detect_mouse = 1;
|
|
|
|
}
|
|
|
|
if ((BX_SER_THIS detect_mouse == 1) && new_b1) {
|
|
|
|
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);
|
|
|
|
}
|
2013-02-23 19:15:59 +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) {
|
2014-01-26 01:05:49 +04:00
|
|
|
BX_DEBUG(("com%d: mouse detection mode", port+1));
|
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';
|
|
|
|
}
|
2014-01-30 20:54:52 +04:00
|
|
|
bx_pc_system.activate_timer(BX_SER_THIS s[port].rx_timer_index,
|
|
|
|
BX_SER_THIS s[port].databyte_usec,
|
|
|
|
0); /* not continuous */
|
2004-12-05 23:23:39 +03:00
|
|
|
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
|
2013-02-23 19:15:59 +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) {
|
2016-11-23 19:42:22 +03:00
|
|
|
if (!BX_SER_THIS s[port].modem_cntl.local_loopback) {
|
|
|
|
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,
|
2014-01-26 01:05:49 +04:00
|
|
|
BX_SER_THIS s[port].databyte_usec * 3,
|
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;
|
2014-01-26 01:05:49 +04:00
|
|
|
Bit8u port = (Bit8u)bx_pc_system.triggeredTimerParam();
|
2013-12-01 22:26:37 +04:00
|
|
|
char pname[20];
|
2004-01-17 18:51:09 +03:00
|
|
|
|
2016-11-23 19:42:22 +03:00
|
|
|
switch (BX_SER_THIS s[port].io_mode) {
|
|
|
|
case BX_SER_MODE_FILE:
|
|
|
|
if (BX_SER_THIS s[port].output == NULL) {
|
|
|
|
sprintf(pname, "ports.serial.%d", port+1);
|
|
|
|
bx_list_c *base = (bx_list_c*) SIM->get_param(pname);
|
|
|
|
bx_param_string_c *devparam = SIM->get_param_string("dev", base);
|
|
|
|
if (!devparam->isempty()) {
|
|
|
|
BX_SER_THIS s[port].output = fopen(devparam->getptr(), "wb");
|
|
|
|
}
|
2013-12-01 22:26:37 +04:00
|
|
|
if (BX_SER_THIS s[port].output == NULL) {
|
2016-11-23 19:42:22 +03:00
|
|
|
BX_ERROR(("Could not open '%s' to write com%d output",
|
|
|
|
devparam->getptr(), port+1));
|
|
|
|
BX_SER_THIS s[port].io_mode = BX_SER_MODE_NULL;
|
2013-12-01 22:26:37 +04:00
|
|
|
}
|
2016-11-23 19:42:22 +03:00
|
|
|
}
|
|
|
|
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:
|
2004-07-28 23:36:42 +04:00
|
|
|
#if defined(SERIAL_ENABLE)
|
2016-11-23 19:42:22 +03:00
|
|
|
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);
|
|
|
|
}
|
2004-07-28 23:36:42 +04:00
|
|
|
#endif
|
2016-11-23 19:42:22 +03:00
|
|
|
break;
|
|
|
|
case BX_SER_MODE_RAW:
|
2003-10-12 14:51:58 +04:00
|
|
|
#if USE_RAW_SERIAL
|
2016-11-23 19:42:22 +03: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
|
2016-11-23 19:42:22 +03:00
|
|
|
break;
|
|
|
|
case BX_SER_MODE_MOUSE:
|
|
|
|
BX_INFO(("com%d: write to mouse ignored: 0x%02x", port+1, BX_SER_THIS s[port].tsrbuffer));
|
|
|
|
break;
|
|
|
|
case BX_SER_MODE_SOCKET_CLIENT:
|
|
|
|
case BX_SER_MODE_SOCKET_SERVER:
|
|
|
|
if (BX_SER_THIS s[port].socket_id >= 0) {
|
2013-11-01 22:19:52 +04:00
|
|
|
#ifdef BX_SER_WIN32
|
2016-11-23 19:42:22 +03:00
|
|
|
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);
|
2005-07-10 20:51:09 +04:00
|
|
|
#else
|
2016-11-23 19:42:22 +03:00
|
|
|
::write(BX_SER_THIS s[port].socket_id,
|
|
|
|
(bx_ptr_t) & BX_SER_THIS s[port].tsrbuffer, 1);
|
2005-07-10 20:51:09 +04:00
|
|
|
#endif
|
2016-11-23 19:42:22 +03:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
case BX_SER_MODE_PIPE_CLIENT:
|
|
|
|
case BX_SER_MODE_PIPE_SERVER:
|
2013-11-01 22:19:52 +04:00
|
|
|
#ifdef BX_SER_WIN32
|
2016-11-23 19:42:22 +03:00
|
|
|
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);
|
|
|
|
}
|
2008-05-22 12:13:22 +04:00
|
|
|
#endif
|
2016-11-23 19:42:22 +03:00
|
|
|
break;
|
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;
|
2014-12-16 05:26:00 +03:00
|
|
|
memmove(&BX_SER_THIS s[port].tx_fifo[0], &BX_SER_THIS s[port].tx_fifo[1], 15);
|
2004-01-17 18:51:09 +03:00
|
|
|
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,
|
2014-01-26 01:05:49 +04:00
|
|
|
BX_SER_THIS s[port].databyte_usec,
|
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
|
2014-01-26 01:05:49 +04:00
|
|
|
Bit8u port = (Bit8u)bx_pc_system.triggeredTimerParam();
|
2004-07-28 23:36:42 +04:00
|
|
|
bx_bool data_ready = 0;
|
2014-01-26 01:05:49 +04:00
|
|
|
int db_usec = BX_SER_THIS s[port].databyte_usec;
|
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) {
|
2013-02-23 19:15:59 +04:00
|
|
|
case BX_SER_MODE_SOCKET_CLIENT:
|
|
|
|
case BX_SER_MODE_SOCKET_SERVER:
|
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)
|
2013-11-01 22:19:52 +04:00
|
|
|
#ifdef BX_SER_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:
|
2014-01-26 17:48:10 +04:00
|
|
|
if (BX_SER_THIS mouse_update && (BX_SER_THIS mouse_internal_buffer.num_elements == 0)) {
|
|
|
|
BX_SER_THIS update_mouse_data();
|
|
|
|
}
|
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;
|
2013-02-23 19:15:59 +04:00
|
|
|
case BX_SER_MODE_PIPE_CLIENT:
|
|
|
|
case BX_SER_MODE_PIPE_SERVER:
|
2013-11-01 22:19:52 +04:00
|
|
|
#ifdef BX_SER_WIN32
|
2008-05-22 12:13:22 +04:00
|
|
|
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) {
|
2014-01-26 01:05:49 +04:00
|
|
|
db_usec = 100000; // Poll frequency is 100ms
|
2003-11-09 03:14:43 +03:00
|
|
|
}
|
2001-04-10 05:04:59 +04:00
|
|
|
}
|
|
|
|
} else {
|
|
|
|
// Poll at 4x baud rate to see if the next-char can
|
|
|
|
// be read
|
2014-01-26 01:05:49 +04:00
|
|
|
db_usec *= 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,
|
2014-01-26 01:05:49 +04:00
|
|
|
db_usec, 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
|
|
|
{
|
2014-01-26 01:05:49 +04:00
|
|
|
Bit8u port = (Bit8u)bx_pc_system.triggeredTimerParam();
|
2014-01-17 22:25:13 +04:00
|
|
|
|
2004-01-17 18:51:09 +03:00
|
|
|
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
|
|
|
{
|
|
|
|
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;
|
|
|
|
|
2014-01-26 17:48:10 +04:00
|
|
|
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;
|
|
|
|
BX_SER_THIS mouse_delayed_dz = delta_z;
|
|
|
|
BX_SER_THIS mouse_buttons = button_state;
|
|
|
|
BX_SER_THIS mouse_update = 1;
|
|
|
|
}
|
2004-12-03 00:34:26 +03:00
|
|
|
|
2014-01-26 17:48:10 +04:00
|
|
|
void bx_serial_c::update_mouse_data()
|
|
|
|
{
|
|
|
|
int delta_x, delta_y;
|
|
|
|
Bit8u b1, b2, b3, button_state, mouse_data[5];
|
|
|
|
int bytes, tail;
|
2004-12-03 00:34:26 +03:00
|
|
|
|
|
|
|
|
|
|
|
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;
|
|
|
|
}
|
2014-01-26 17:48:10 +04:00
|
|
|
button_state = BX_SER_THIS mouse_buttons;
|
2004-12-03 00:34:26 +03:00
|
|
|
|
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;
|
2014-01-26 17:48:10 +04:00
|
|
|
b3 = (Bit8u) -((Bit8s) BX_SER_THIS mouse_delayed_dz);
|
2006-12-31 14:56:14 +03:00
|
|
|
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++;
|
|
|
|
}
|
2014-01-26 17:48:10 +04:00
|
|
|
BX_SER_THIS mouse_update = 0;
|
2004-12-03 00:34:26 +03:00
|
|
|
}
|