2001-10-03 17:10:38 +04:00
|
|
|
/////////////////////////////////////////////////////////////////////////
|
2003-10-31 20:23:56 +03:00
|
|
|
// $Id: serial.cc,v 1.40 2003-10-31 17:23:56 vruppert Exp $
|
2001-10-03 17:10:38 +04:00
|
|
|
/////////////////////////////////////////////////////////////////////////
|
|
|
|
//
|
2002-01-08 23:31:14 +03:00
|
|
|
// Copyright (C) 2002 MandrakeSoft S.A.
|
2001-04-10 05:04:59 +04:00
|
|
|
//
|
|
|
|
// MandrakeSoft S.A.
|
|
|
|
// 43, rue d'Aboukir
|
|
|
|
// 75002 Paris - France
|
|
|
|
// http://www.linux-mandrake.com/
|
|
|
|
// http://www.mandrakesoft.com/
|
|
|
|
//
|
|
|
|
// This library is free software; you can redistribute it and/or
|
|
|
|
// modify it under the terms of the GNU Lesser General Public
|
|
|
|
// License as published by the Free Software Foundation; either
|
|
|
|
// version 2 of the License, or (at your option) any later version.
|
|
|
|
//
|
|
|
|
// This library is distributed in the hope that it will be useful,
|
|
|
|
// but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
|
|
// Lesser General Public License for more details.
|
|
|
|
//
|
|
|
|
// You should have received a copy of the GNU Lesser General Public
|
|
|
|
// License along with this library; if not, write to the Free Software
|
|
|
|
// Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
|
|
|
|
|
|
|
|
|
|
|
// Peter Grehan (grehan@iprg.nokia.com) coded most of this
|
|
|
|
// serial emulation. Here's some notes from his implementation:
|
|
|
|
|
2003-10-12 14:51:58 +04:00
|
|
|
// Implement a single 16450, and allow terminal input/output to stdout
|
|
|
|
// on some platforms. 16550/16550a/16552 should be easy: just add code
|
2001-04-10 05:04:59 +04:00
|
|
|
// to handle a FIFO. I only did what was needed to get console output.
|
|
|
|
|
|
|
|
|
2002-10-25 01:07:56 +04:00
|
|
|
// Define BX_PLUGGABLE in files that can be compiled into plugins. For
|
|
|
|
// platforms that require a special tag on exported symbols, BX_PLUGGABLE
|
|
|
|
// is used to know when we are exporting symbols and when we are importing.
|
|
|
|
#define BX_PLUGGABLE
|
|
|
|
|
2001-04-10 05:04:59 +04:00
|
|
|
#include "bochs.h"
|
2002-10-25 01:07:56 +04:00
|
|
|
|
|
|
|
#define LOG_THIS theSerialDevice->
|
2001-04-10 05:04:59 +04:00
|
|
|
|
|
|
|
#if USE_RAW_SERIAL
|
|
|
|
#include <signal.h>
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#ifdef WIN32
|
|
|
|
#ifndef __MINGW32__
|
|
|
|
// +++
|
|
|
|
//#include <winsock2.h>
|
|
|
|
#include <winsock.h>
|
|
|
|
#endif
|
|
|
|
#endif
|
|
|
|
|
2003-10-24 15:16:25 +04:00
|
|
|
#if defined(__NetBSD__) || defined(__FreeBSD__) || defined(__OpenBSD__) || defined(__linux__) || defined(__GNU__) || defined(__APPLE__)
|
2002-03-03 09:03:29 +03:00
|
|
|
#define SERIAL_ENABLE
|
2001-06-19 21:52:46 +04:00
|
|
|
#endif
|
2001-04-10 05:04:59 +04:00
|
|
|
|
2001-06-19 21:52:46 +04:00
|
|
|
#ifdef SERIAL_ENABLE
|
2001-04-10 05:04:59 +04:00
|
|
|
extern "C" {
|
|
|
|
#include <termios.h>
|
|
|
|
};
|
|
|
|
#endif
|
|
|
|
|
2001-06-19 21:52:46 +04:00
|
|
|
#ifdef SERIAL_ENABLE
|
2001-04-10 05:04:59 +04:00
|
|
|
static struct termios term_orig, term_new;
|
|
|
|
#endif
|
|
|
|
|
|
|
|
static int tty_id;
|
|
|
|
|
2002-10-25 01:07:56 +04:00
|
|
|
bx_serial_c *theSerialDevice = NULL;
|
|
|
|
|
|
|
|
int
|
|
|
|
libserial_LTX_plugin_init(plugin_t *plugin, plugintype_t type, int argc, char *argv[])
|
|
|
|
{
|
|
|
|
theSerialDevice = new bx_serial_c ();
|
|
|
|
bx_devices.pluginSerialDevice = theSerialDevice;
|
|
|
|
BX_REGISTER_DEVICE_DEVMODEL(plugin, type, theSerialDevice, BX_PLUGIN_SERIAL);
|
|
|
|
return(0); // Success
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
libserial_LTX_plugin_fini(void)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2001-04-10 05:04:59 +04:00
|
|
|
bx_serial_c::bx_serial_c(void)
|
|
|
|
{
|
2001-06-27 23:16:01 +04:00
|
|
|
put("SER");
|
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
|
|
|
settype(SERLOG);
|
2002-08-24 21:11:33 +04:00
|
|
|
tty_id = -1;
|
2002-10-06 23:04:47 +04:00
|
|
|
for (int i=0; i<BX_SERIAL_MAXDEV; i++) {
|
|
|
|
s[i].tx_timer_index = BX_NULL_TIMER_HANDLE;
|
|
|
|
s[i].rx_timer_index = BX_NULL_TIMER_HANDLE;
|
|
|
|
}
|
2002-03-03 09:03:29 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
bx_serial_c::~bx_serial_c(void)
|
|
|
|
{
|
|
|
|
#ifdef SERIAL_ENABLE
|
2002-08-24 21:11:33 +04:00
|
|
|
if ((bx_options.com[0].Oenabled->get ()) && (tty_id >= 0))
|
2002-06-05 01:35:08 +04:00
|
|
|
tcsetattr(tty_id, TCSAFLUSH, &term_orig);
|
2002-03-03 09:03:29 +03:00
|
|
|
#endif
|
|
|
|
// nothing for now
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
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};
|
|
|
|
char name[16];
|
|
|
|
|
2002-08-24 21:11:33 +04:00
|
|
|
if (!bx_options.com[0].Oenabled->get ())
|
2002-03-03 09:03:29 +03:00
|
|
|
return;
|
|
|
|
|
2001-06-19 21:52:46 +04:00
|
|
|
#ifdef SERIAL_ENABLE
|
2002-08-24 23:03:43 +04:00
|
|
|
if (strlen(bx_options.com[0].Odev->getptr ()) > 0) {
|
|
|
|
tty_id = open(bx_options.com[0].Odev->getptr (), O_RDWR|O_NONBLOCK,600);
|
|
|
|
if (tty_id < 0)
|
|
|
|
BX_PANIC(("open of %s (%s) failed\n",
|
|
|
|
"com1", bx_options.com[0].Odev->getptr ()));
|
|
|
|
BX_DEBUG(("tty_id: %d",tty_id));
|
|
|
|
tcgetattr(tty_id, &term_orig);
|
|
|
|
bcopy((caddr_t) &term_orig, (caddr_t) &term_new, sizeof(struct termios));
|
|
|
|
cfmakeraw(&term_new);
|
|
|
|
term_new.c_oflag |= OPOST | ONLCR; // Enable NL to CR-NL translation
|
2001-04-10 05:04:59 +04:00
|
|
|
#ifndef TRUE_CTLC
|
2002-08-24 23:03:43 +04:00
|
|
|
// ctl-C will exit Bochs, or trap to the debugger
|
|
|
|
term_new.c_iflag &= ~IGNBRK;
|
|
|
|
term_new.c_iflag |= BRKINT;
|
|
|
|
term_new.c_lflag |= ISIG;
|
2001-04-10 05:04:59 +04:00
|
|
|
#else
|
2002-08-24 23:03:43 +04:00
|
|
|
// ctl-C will be delivered to the serial port
|
|
|
|
term_new.c_iflag |= IGNBRK;
|
|
|
|
term_new.c_iflag &= ~BRKINT;
|
2001-05-17 10:44:06 +04:00
|
|
|
#endif /* !def TRUE_CTLC */
|
2002-08-24 23:03:43 +04:00
|
|
|
term_new.c_iflag = 0;
|
|
|
|
term_new.c_oflag = 0;
|
|
|
|
term_new.c_cflag = CS8|CREAD|CLOCAL;
|
|
|
|
term_new.c_lflag = 0;
|
|
|
|
term_new.c_cc[VMIN] = 1;
|
|
|
|
term_new.c_cc[VTIME] = 0;
|
|
|
|
//term_new.c_iflag |= IXOFF;
|
|
|
|
tcsetattr(tty_id, TCSAFLUSH, &term_new);
|
|
|
|
}
|
2001-06-19 21:52:46 +04:00
|
|
|
#endif /* def SERIAL_ENABLE */
|
2001-04-10 05:04:59 +04:00
|
|
|
// nothing for now
|
|
|
|
#if USE_RAW_SERIAL
|
2001-06-15 16:32:15 +04:00
|
|
|
this->raw = new serial_raw("/dev/cua0", SIGUSR1);
|
2001-04-10 05:04:59 +04:00
|
|
|
#endif // USE_RAW_SERIAL
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Put the UART registers into their RESET state
|
|
|
|
*/
|
2003-10-30 00:00:04 +03:00
|
|
|
for (unsigned i=0; i<BX_N_SERIAL_PORTS; i++) {
|
|
|
|
if (bx_options.com[i].Oenabled->get ()) {
|
|
|
|
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;
|
|
|
|
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;
|
|
|
|
|
|
|
|
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
|
|
|
|
}
|
|
|
|
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;
|
|
|
|
BX_SER_THIS s[i].int_ident.fifo_enabled = 0;
|
|
|
|
|
|
|
|
/* FIFO control: b0000 0000 */
|
|
|
|
BX_SER_THIS s[i].fifo_cntl.enable = 0;
|
|
|
|
BX_SER_THIS s[i].fifo_cntl.rxreset = 0;
|
|
|
|
BX_SER_THIS s[i].fifo_cntl.txreset = 0;
|
|
|
|
BX_SER_THIS s[i].fifo_cntl.dmamode = 0;
|
|
|
|
BX_SER_THIS s[i].fifo_cntl.rxtrigger = 0;
|
|
|
|
|
|
|
|
/* 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++) {
|
|
|
|
BX_DEBUG(("register read/write: 0x%04x",addr));
|
|
|
|
DEV_register_ioread_handler(this, read_handler, addr, name, 1);
|
|
|
|
DEV_register_iowrite_handler(this, write_handler, addr, name, 1);
|
|
|
|
}
|
|
|
|
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
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2002-08-27 23:54:46 +04:00
|
|
|
void
|
|
|
|
bx_serial_c::reset(unsigned type)
|
|
|
|
{
|
|
|
|
}
|
2001-04-10 05:04:59 +04:00
|
|
|
|
2003-10-30 00:00:04 +03:00
|
|
|
void
|
|
|
|
bx_serial_c::lower_interrupt(Bit8u port)
|
|
|
|
{
|
|
|
|
/* 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) &&
|
|
|
|
(BX_SER_THIS s[port].ms_interrupt == 0)) {
|
|
|
|
DEV_pic_lower_irq(BX_SER_THIS s[port].IRQ);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2003-10-31 20:23:56 +03:00
|
|
|
void
|
|
|
|
bx_serial_c::raise_interrupt(Bit8u port, int type)
|
|
|
|
{
|
|
|
|
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;
|
|
|
|
}
|
|
|
|
if (gen_int && BX_SER_THIS s[port].modem_cntl.out2) {
|
|
|
|
DEV_pic_raise_irq(BX_SER_THIS s[port].IRQ);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2001-04-10 05:04:59 +04:00
|
|
|
// static IO port read callback handler
|
|
|
|
// redirects to non-static class handler to avoid virtual functions
|
|
|
|
|
|
|
|
Bit32u
|
|
|
|
bx_serial_c::read_handler(void *this_ptr, Bit32u address, unsigned io_len)
|
|
|
|
{
|
|
|
|
#if !BX_USE_SER_SMF
|
|
|
|
bx_serial_c *class_ptr = (bx_serial_c *) this_ptr;
|
|
|
|
|
|
|
|
return( class_ptr->read(address, io_len) );
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
Bit32u
|
|
|
|
bx_serial_c::read(Bit32u address, unsigned io_len)
|
|
|
|
{
|
|
|
|
#else
|
|
|
|
UNUSED(this_ptr);
|
|
|
|
#endif // !BX_USE_SER_SMF
|
2001-06-19 21:52:46 +04:00
|
|
|
//UNUSED(address);
|
2001-04-10 05:04:59 +04:00
|
|
|
Bit8u val;
|
|
|
|
|
|
|
|
/* SERIAL PORT 1 */
|
|
|
|
|
2002-08-27 21:24:36 +04:00
|
|
|
BX_DEBUG(("register read from address 0x%04x - ", (unsigned) address));
|
2001-04-10 05:04:59 +04:00
|
|
|
|
|
|
|
switch (address) {
|
|
|
|
case 0x03F8: /* receive buffer, or divisor latch LSB if DLAB set */
|
|
|
|
if (BX_SER_THIS s[0].line_cntl.dlab) {
|
2002-01-20 19:35:32 +03:00
|
|
|
val = BX_SER_THIS s[0].divisor_lsb;
|
2001-04-10 05:04:59 +04:00
|
|
|
} else {
|
2002-01-20 19:35:32 +03:00
|
|
|
val = BX_SER_THIS s[0].rxbuffer;
|
|
|
|
BX_SER_THIS s[0].line_status.rxdata_ready = 0;
|
|
|
|
BX_SER_THIS s[0].rx_interrupt = 0;
|
|
|
|
BX_SER_THIS s[0].rx_ipending = 0;
|
2003-10-30 00:00:04 +03:00
|
|
|
lower_interrupt(0);
|
2001-04-10 05:04:59 +04:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case 0x03F9: /* interrupt enable register, or div. latch MSB */
|
|
|
|
if (BX_SER_THIS s[0].line_cntl.dlab) {
|
2002-01-20 19:35:32 +03:00
|
|
|
val = BX_SER_THIS s[0].divisor_msb;
|
2001-04-10 05:04:59 +04:00
|
|
|
} else {
|
2002-01-20 19:35:32 +03:00
|
|
|
val = BX_SER_THIS s[0].int_enable.rxdata_enable |
|
|
|
|
(BX_SER_THIS s[0].int_enable.txhold_enable << 1) |
|
|
|
|
(BX_SER_THIS s[0].int_enable.rxlstat_enable << 2) |
|
|
|
|
(BX_SER_THIS s[0].int_enable.modstat_enable << 3);
|
2001-04-10 05:04:59 +04:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case 0x03FA: /* interrupt ID register */
|
|
|
|
/*
|
|
|
|
* Set the interrupt ID based on interrupt source
|
|
|
|
*/
|
2002-01-20 19:35:32 +03:00
|
|
|
if (BX_SER_THIS s[0].ls_interrupt) {
|
|
|
|
BX_SER_THIS s[0].int_ident.int_ID = 0x3;
|
|
|
|
BX_SER_THIS s[0].int_ident.ipending = 0;
|
|
|
|
} else if (BX_SER_THIS s[0].rx_interrupt) {
|
|
|
|
BX_SER_THIS s[0].int_ident.int_ID = 0x2;
|
|
|
|
BX_SER_THIS s[0].int_ident.ipending = 0;
|
2001-04-10 05:04:59 +04:00
|
|
|
} else if (BX_SER_THIS s[0].tx_interrupt) {
|
2002-01-20 19:35:32 +03:00
|
|
|
BX_SER_THIS s[0].int_ident.int_ID = 0x1;
|
|
|
|
BX_SER_THIS s[0].int_ident.ipending = 0;
|
|
|
|
} else if (BX_SER_THIS s[0].ms_interrupt) {
|
|
|
|
BX_SER_THIS s[0].int_ident.int_ID = 0x0;
|
|
|
|
BX_SER_THIS s[0].int_ident.ipending = 0;
|
2001-04-10 05:04:59 +04:00
|
|
|
} else {
|
2002-01-20 19:35:32 +03:00
|
|
|
BX_SER_THIS s[0].int_ident.int_ID = 0x0;
|
|
|
|
BX_SER_THIS s[0].int_ident.ipending = 1;
|
|
|
|
}
|
2001-04-10 05:04:59 +04:00
|
|
|
BX_SER_THIS s[0].tx_interrupt = 0;
|
2003-10-30 00:00:04 +03:00
|
|
|
lower_interrupt(0);
|
2001-04-10 05:04:59 +04:00
|
|
|
|
|
|
|
val = BX_SER_THIS s[0].int_ident.ipending |
|
2002-01-20 19:35:32 +03:00
|
|
|
(BX_SER_THIS s[0].int_ident.int_ID << 1) |
|
|
|
|
(BX_SER_THIS s[0].int_ident.fifo_enabled << 6);
|
2001-04-10 05:04:59 +04:00
|
|
|
break;
|
|
|
|
|
|
|
|
case 0x03FB: /* Line control register */
|
|
|
|
val = BX_SER_THIS s[0].line_cntl.wordlen_sel |
|
|
|
|
(BX_SER_THIS s[0].line_cntl.stopbits << 2) |
|
|
|
|
(BX_SER_THIS s[0].line_cntl.parity_enable << 3) |
|
|
|
|
(BX_SER_THIS s[0].line_cntl.evenparity_sel << 4) |
|
|
|
|
(BX_SER_THIS s[0].line_cntl.stick_parity << 5) |
|
|
|
|
(BX_SER_THIS s[0].line_cntl.break_cntl << 6) |
|
|
|
|
(BX_SER_THIS s[0].line_cntl.dlab << 7);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case 0x03FC: /* MODEM control register */
|
2002-01-08 23:31:14 +03:00
|
|
|
val = BX_SER_THIS s[0].modem_cntl.dtr |
|
|
|
|
(BX_SER_THIS s[0].modem_cntl.rts << 1) |
|
|
|
|
(BX_SER_THIS s[0].modem_cntl.out1 << 2) |
|
|
|
|
(BX_SER_THIS s[0].modem_cntl.out2 << 3) |
|
|
|
|
(BX_SER_THIS s[0].modem_cntl.local_loopback << 4);
|
2001-04-10 05:04:59 +04:00
|
|
|
break;
|
|
|
|
|
|
|
|
case 0x03FD: /* Line status register */
|
|
|
|
val = BX_SER_THIS s[0].line_status.rxdata_ready |
|
|
|
|
(BX_SER_THIS s[0].line_status.overrun_error << 1) |
|
|
|
|
(BX_SER_THIS s[0].line_status.parity_error << 2) |
|
|
|
|
(BX_SER_THIS s[0].line_status.framing_error << 3) |
|
|
|
|
(BX_SER_THIS s[0].line_status.break_int << 4) |
|
2003-09-15 00:16:25 +04:00
|
|
|
(BX_SER_THIS s[0].line_status.thr_empty << 5) |
|
|
|
|
(BX_SER_THIS s[0].line_status.tsr_empty << 6) |
|
2001-04-10 05:04:59 +04:00
|
|
|
(BX_SER_THIS s[0].line_status.fifo_error << 7);
|
2002-01-20 19:35:32 +03:00
|
|
|
BX_SER_THIS s[0].line_status.overrun_error = 0;
|
2001-04-10 05:04:59 +04:00
|
|
|
BX_SER_THIS s[0].line_status.break_int = 0;
|
2002-01-20 19:35:32 +03:00
|
|
|
BX_SER_THIS s[0].ls_interrupt = 0;
|
|
|
|
BX_SER_THIS s[0].ls_ipending = 0;
|
2003-10-30 00:00:04 +03:00
|
|
|
lower_interrupt(0);
|
2001-04-10 05:04:59 +04:00
|
|
|
break;
|
|
|
|
|
|
|
|
case 0x03FE: /* MODEM status register */
|
|
|
|
val = BX_SER_THIS s[0].modem_status.delta_cts |
|
|
|
|
(BX_SER_THIS s[0].modem_status.delta_dsr << 1) |
|
|
|
|
(BX_SER_THIS s[0].modem_status.ri_trailedge << 2) |
|
|
|
|
(BX_SER_THIS s[0].modem_status.delta_dcd << 3) |
|
|
|
|
(BX_SER_THIS s[0].modem_status.cts << 4) |
|
|
|
|
(BX_SER_THIS s[0].modem_status.dsr << 5) |
|
|
|
|
(BX_SER_THIS s[0].modem_status.ri << 6) |
|
|
|
|
(BX_SER_THIS s[0].modem_status.dcd << 7);
|
2001-11-17 21:10:54 +03:00
|
|
|
BX_SER_THIS s[0].modem_status.delta_cts = 0;
|
|
|
|
BX_SER_THIS s[0].modem_status.delta_dsr = 0;
|
|
|
|
BX_SER_THIS s[0].modem_status.ri_trailedge = 0;
|
|
|
|
BX_SER_THIS s[0].modem_status.delta_dcd = 0;
|
2002-01-20 19:35:32 +03:00
|
|
|
BX_SER_THIS s[0].ms_interrupt = 0;
|
|
|
|
BX_SER_THIS s[0].ms_ipending = 0;
|
2003-10-30 00:00:04 +03:00
|
|
|
lower_interrupt(0);
|
2001-04-10 05:04:59 +04:00
|
|
|
break;
|
|
|
|
|
|
|
|
case 0x03FF: /* scratch register */
|
2001-11-17 21:10:54 +03:00
|
|
|
val = BX_SER_THIS s[0].scratch;
|
2001-04-10 05:04:59 +04:00
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
val = 0; // keep compiler happy
|
2002-08-27 21:24:36 +04:00
|
|
|
BX_PANIC(("unsupported io read from address=0x%04x!",
|
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));
|
2001-04-10 05:04:59 +04:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2002-08-27 21:24:36 +04:00
|
|
|
BX_DEBUG(("val = 0x%02x", (unsigned) val));
|
2001-04-10 05:04:59 +04:00
|
|
|
|
|
|
|
return(val);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// static IO port write callback handler
|
|
|
|
// redirects to non-static class handler to avoid virtual functions
|
|
|
|
|
2001-06-19 21:52:46 +04:00
|
|
|
void
|
2001-04-10 05:04:59 +04:00
|
|
|
bx_serial_c::write_handler(void *this_ptr, Bit32u address, Bit32u value, unsigned io_len)
|
|
|
|
{
|
|
|
|
#if !BX_USE_SER_SMF
|
|
|
|
bx_serial_c *class_ptr = (bx_serial_c *) this_ptr;
|
|
|
|
|
|
|
|
class_ptr->write(address, value, io_len);
|
|
|
|
}
|
|
|
|
|
2001-06-19 21:52:46 +04:00
|
|
|
void
|
2001-04-10 05:04:59 +04:00
|
|
|
bx_serial_c::write(Bit32u address, Bit32u value, unsigned io_len)
|
|
|
|
{
|
|
|
|
#else
|
|
|
|
UNUSED(this_ptr);
|
|
|
|
#endif // !BX_USE_SER_SMF
|
2002-10-25 15:44:41 +04:00
|
|
|
bx_bool prev_cts, prev_dsr, prev_ri, prev_dcd;
|
2003-10-28 21:40:00 +03:00
|
|
|
bx_bool new_rx_ien, new_tx_ien, new_ls_ien, new_ms_ien;
|
2002-10-25 15:44:41 +04:00
|
|
|
bx_bool gen_int = 0;
|
2002-01-20 19:35:32 +03:00
|
|
|
|
2001-04-10 05:04:59 +04:00
|
|
|
/* SERIAL PORT 1 */
|
|
|
|
|
2002-08-27 21:24:36 +04:00
|
|
|
BX_DEBUG(("write to address: 0x%04x = 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
|
|
|
|
|
|
|
switch (address) {
|
|
|
|
case 0x03F8: /* transmit buffer, or divisor latch LSB if DLAB set */
|
2002-01-20 19:35:32 +03:00
|
|
|
if (BX_SER_THIS s[0].line_cntl.dlab) {
|
2001-04-10 05:04:59 +04:00
|
|
|
BX_SER_THIS s[0].divisor_lsb = value;
|
|
|
|
|
2003-10-31 20:23:56 +03:00
|
|
|
if ((value != 0) || (BX_SER_THIS s[0].divisor_msb != 0)) {
|
2001-04-10 05:04:59 +04:00
|
|
|
BX_SER_THIS s[0].baudrate = (int) (BX_PC_CLOCK_XTL /
|
|
|
|
(16 * ((BX_SER_THIS s[0].divisor_msb << 8) |
|
|
|
|
BX_SER_THIS s[0].divisor_lsb)));
|
|
|
|
#if USE_RAW_SERIAL
|
|
|
|
BX_SER_THIS raw->set_baudrate(BX_SER_THIS s[0].baudrate);
|
|
|
|
#endif // USE_RAW_SERIAL
|
|
|
|
}
|
|
|
|
} else {
|
2003-09-15 00:16:25 +04:00
|
|
|
if (BX_SER_THIS s[0].line_status.thr_empty) {
|
2002-01-20 19:35:32 +03:00
|
|
|
Bit8u bitmask = 0xff >> (3 - BX_SER_THIS s[0].line_cntl.wordlen_sel);
|
2003-09-15 00:16:25 +04:00
|
|
|
BX_SER_THIS s[0].thrbuffer = value & bitmask;
|
|
|
|
if (BX_SER_THIS s[0].line_status.tsr_empty) {
|
|
|
|
BX_SER_THIS s[0].tsrbuffer = BX_SER_THIS s[0].thrbuffer;
|
|
|
|
BX_SER_THIS s[0].line_status.tsr_empty = 0;
|
2003-10-31 20:23:56 +03:00
|
|
|
raise_interrupt(0, BX_SER_INT_TXHOLD);
|
2003-09-15 00:16:25 +04:00
|
|
|
bx_pc_system.activate_timer(BX_SER_THIS s[0].tx_timer_index,
|
|
|
|
(int) (1000000.0 / BX_SER_THIS s[0].baudrate *
|
|
|
|
(BX_SER_THIS s[0].line_cntl.wordlen_sel + 5)),
|
|
|
|
0); /* not continuous */
|
|
|
|
} else {
|
|
|
|
BX_SER_THIS s[0].line_status.thr_empty = 0;
|
|
|
|
BX_SER_THIS s[0].tx_interrupt = 0;
|
2003-10-30 00:00:04 +03:00
|
|
|
lower_interrupt(0);
|
2003-09-15 00:16:25 +04:00
|
|
|
}
|
2002-01-20 19:35:32 +03:00
|
|
|
} else {
|
|
|
|
BX_ERROR(("write to tx hold register when not empty"));
|
|
|
|
}
|
2001-04-10 05:04:59 +04:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case 0x03F9: /* interrupt enable register, or div. latch MSB */
|
|
|
|
if (BX_SER_THIS s[0].line_cntl.dlab) {
|
|
|
|
BX_SER_THIS s[0].divisor_msb = value;
|
|
|
|
|
2003-10-31 20:23:56 +03:00
|
|
|
if ((value != 0) || (BX_SER_THIS s[0].divisor_lsb != 0)) {
|
2001-04-10 05:04:59 +04:00
|
|
|
BX_SER_THIS s[0].baudrate = (int) (BX_PC_CLOCK_XTL /
|
|
|
|
(16 * ((BX_SER_THIS s[0].divisor_msb << 8) |
|
|
|
|
BX_SER_THIS s[0].divisor_lsb)));
|
|
|
|
#if USE_RAW_SERIAL
|
|
|
|
BX_SER_THIS raw->set_baudrate(BX_SER_THIS s[0].baudrate);
|
|
|
|
#endif // USE_RAW_SERIAL
|
|
|
|
}
|
|
|
|
} else {
|
2003-10-28 21:40:00 +03:00
|
|
|
new_rx_ien = value & 0x01;
|
|
|
|
new_tx_ien = (value & 0x02) >> 1;
|
|
|
|
new_ls_ien = (value & 0x04) >> 2;
|
|
|
|
new_ms_ien = (value & 0x08) >> 3;
|
|
|
|
if (new_ms_ien != BX_SER_THIS s[0].int_enable.modstat_enable) {
|
|
|
|
BX_SER_THIS s[0].int_enable.modstat_enable = new_ms_ien;
|
2003-10-30 00:00:04 +03:00
|
|
|
if (BX_SER_THIS s[0].int_enable.modstat_enable == 1) {
|
|
|
|
if (BX_SER_THIS s[0].ms_ipending == 1) {
|
|
|
|
BX_SER_THIS s[0].ms_interrupt = 1;
|
|
|
|
BX_SER_THIS s[0].ms_ipending = 0;
|
|
|
|
gen_int = 1;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
if (BX_SER_THIS s[0].ms_interrupt == 1) {
|
|
|
|
BX_SER_THIS s[0].ms_interrupt = 0;
|
|
|
|
BX_SER_THIS s[0].ms_ipending = 1;
|
|
|
|
lower_interrupt(0);
|
|
|
|
}
|
2003-10-28 21:40:00 +03:00
|
|
|
}
|
2002-01-20 19:35:32 +03:00
|
|
|
}
|
2003-10-28 21:40:00 +03:00
|
|
|
if (new_tx_ien != BX_SER_THIS s[0].int_enable.txhold_enable) {
|
|
|
|
BX_SER_THIS s[0].int_enable.txhold_enable = new_tx_ien;
|
|
|
|
if (BX_SER_THIS s[0].int_enable.txhold_enable == 1) {
|
|
|
|
BX_SER_THIS s[0].tx_interrupt = BX_SER_THIS s[0].line_status.thr_empty;
|
|
|
|
if (BX_SER_THIS s[0].tx_interrupt) gen_int = 1;
|
|
|
|
} else {
|
|
|
|
BX_SER_THIS s[0].tx_interrupt = 0;
|
2003-10-30 00:00:04 +03:00
|
|
|
lower_interrupt(0);
|
2003-10-28 21:40:00 +03:00
|
|
|
}
|
2002-01-20 19:35:32 +03:00
|
|
|
}
|
2003-10-28 21:40:00 +03:00
|
|
|
if (new_rx_ien != BX_SER_THIS s[0].int_enable.rxdata_enable) {
|
|
|
|
BX_SER_THIS s[0].int_enable.rxdata_enable = new_rx_ien;
|
2003-10-30 00:00:04 +03:00
|
|
|
if (BX_SER_THIS s[0].int_enable.rxdata_enable == 1) {
|
|
|
|
if (BX_SER_THIS s[0].rx_ipending == 1) {
|
|
|
|
BX_SER_THIS s[0].rx_interrupt = 1;
|
|
|
|
BX_SER_THIS s[0].rx_ipending = 0;
|
|
|
|
gen_int = 1;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
if (BX_SER_THIS s[0].rx_interrupt == 1) {
|
|
|
|
BX_SER_THIS s[0].rx_interrupt = 0;
|
|
|
|
BX_SER_THIS s[0].rx_ipending = 1;
|
|
|
|
lower_interrupt(0);
|
|
|
|
}
|
2003-10-28 21:40:00 +03:00
|
|
|
}
|
2002-01-20 19:35:32 +03:00
|
|
|
}
|
2003-10-28 21:40:00 +03:00
|
|
|
if (new_ls_ien != BX_SER_THIS s[0].int_enable.rxlstat_enable) {
|
|
|
|
BX_SER_THIS s[0].int_enable.rxlstat_enable = new_ls_ien;
|
2003-10-30 00:00:04 +03:00
|
|
|
if (BX_SER_THIS s[0].int_enable.rxlstat_enable == 1) {
|
|
|
|
if (BX_SER_THIS s[0].ls_ipending == 1) {
|
|
|
|
BX_SER_THIS s[0].ls_interrupt = 1;
|
|
|
|
BX_SER_THIS s[0].ls_ipending = 0;
|
|
|
|
gen_int = 1;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
if (BX_SER_THIS s[0].ls_interrupt == 1) {
|
|
|
|
BX_SER_THIS s[0].ls_interrupt = 0;
|
|
|
|
BX_SER_THIS s[0].ls_ipending = 1;
|
|
|
|
lower_interrupt(0);
|
|
|
|
}
|
2003-10-28 21:40:00 +03:00
|
|
|
}
|
2002-01-20 19:35:32 +03:00
|
|
|
}
|
2003-10-31 20:23:56 +03:00
|
|
|
if (gen_int) raise_interrupt(0, BX_SER_INT_IER);
|
2001-04-10 05:04:59 +04:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case 0x03FA: /* FIFO control register */
|
|
|
|
/* Ignore until FIFO mode is supported... */
|
|
|
|
break;
|
|
|
|
|
|
|
|
case 0x03FB: /* Line control register */
|
|
|
|
#if !USE_RAW_SERIAL
|
|
|
|
if ((value & 0x3) != 0x3) {
|
|
|
|
/* ignore this: this is set by FreeBSD when the console
|
|
|
|
code wants to set DLAB */
|
|
|
|
}
|
|
|
|
#endif // !USE_RAW_SERIAL
|
|
|
|
#if USE_RAW_SERIAL
|
|
|
|
if (BX_SER_THIS s[0].line_cntl.wordlen_sel != (value & 0x3)) {
|
|
|
|
BX_SER_THIS raw->set_data_bits((value & 0x3) + 5);
|
|
|
|
}
|
|
|
|
if (BX_SER_THIS s[0].line_cntl.stopbits != (value & 0x4) >> 2) {
|
|
|
|
BX_SER_THIS raw->set_stop_bits((value & 0x4 >> 2) ? 2 : 1);
|
|
|
|
}
|
|
|
|
if (BX_SER_THIS s[0].line_cntl.parity_enable != (value & 0x8) >> 3 ||
|
|
|
|
BX_SER_THIS s[0].line_cntl.evenparity_sel != (value & 0x10) >> 4 ||
|
|
|
|
BX_SER_THIS s[0].line_cntl.stick_parity != (value & 0x20) >> 5) {
|
|
|
|
if (((value & 0x20) >> 5) &&
|
|
|
|
((value & 0x8) >> 3))
|
2001-11-17 21:10:54 +03:00
|
|
|
BX_PANIC(("sticky parity set and parity enabled"));
|
2001-04-10 05:04:59 +04:00
|
|
|
BX_SER_THIS raw->set_parity_mode(((value & 0x8) >> 3),
|
|
|
|
((value & 0x10) >> 4) ? P_EVEN : P_ODD);
|
|
|
|
}
|
|
|
|
if (BX_SER_THIS s[0].line_cntl.break_cntl && !((value & 0x40) >> 6)) {
|
|
|
|
BX_SER_THIS raw->transmit(C_BREAK);
|
|
|
|
}
|
|
|
|
#endif // USE_RAW_SERIAL
|
|
|
|
|
|
|
|
BX_SER_THIS s[0].line_cntl.wordlen_sel = value & 0x3;
|
|
|
|
/* These are ignored, but set them up so they can be read back */
|
|
|
|
BX_SER_THIS s[0].line_cntl.stopbits = (value & 0x4) >> 2;
|
|
|
|
BX_SER_THIS s[0].line_cntl.parity_enable = (value & 0x8) >> 3;
|
|
|
|
BX_SER_THIS s[0].line_cntl.evenparity_sel = (value & 0x10) >> 4;
|
|
|
|
BX_SER_THIS s[0].line_cntl.stick_parity = (value & 0x20) >> 5;
|
|
|
|
BX_SER_THIS s[0].line_cntl.break_cntl = (value & 0x40) >> 6;
|
|
|
|
/* used when doing future writes */
|
|
|
|
if (BX_SER_THIS s[0].line_cntl.dlab &&
|
|
|
|
!((value & 0x80) >> 7)) {
|
|
|
|
// Start the receive polling process if not already started
|
2003-10-12 14:51:58 +04:00
|
|
|
// and there is a valid baudrate.
|
2001-04-10 05:04:59 +04:00
|
|
|
if (BX_SER_THIS s[0].rx_pollstate == BX_SER_RXIDLE &&
|
|
|
|
BX_SER_THIS s[0].baudrate != 0) {
|
|
|
|
BX_SER_THIS s[0].rx_pollstate = BX_SER_RXPOLL;
|
|
|
|
bx_pc_system.activate_timer(BX_SER_THIS s[0].rx_timer_index,
|
2003-10-12 14:51:58 +04:00
|
|
|
(int) (1000000.0 / BX_SER_THIS s[0].baudrate *
|
|
|
|
(BX_SER_THIS s[0].line_cntl.wordlen_sel + 5)),
|
|
|
|
0); /* not continuous */
|
2001-04-10 05:04:59 +04:00
|
|
|
}
|
2001-06-19 21:52:46 +04:00
|
|
|
BX_DEBUG(("baud rate set - %d", BX_SER_THIS s[0].baudrate));
|
2001-04-10 05:04:59 +04:00
|
|
|
}
|
|
|
|
BX_SER_THIS s[0].line_cntl.dlab = (value & 0x80) >> 7;
|
|
|
|
break;
|
2002-01-20 19:35:32 +03:00
|
|
|
|
2001-04-10 05:04:59 +04:00
|
|
|
case 0x03FC: /* MODEM control register */
|
|
|
|
if ((value & 0x01) == 0) {
|
|
|
|
#if USE_RAW_SERIAL
|
|
|
|
BX_SER_THIS raw->send_hangup();
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
BX_SER_THIS s[0].modem_cntl.dtr = value & 0x01;
|
|
|
|
BX_SER_THIS s[0].modem_cntl.rts = (value & 0x02) >> 1;
|
|
|
|
BX_SER_THIS s[0].modem_cntl.out1 = (value & 0x04) >> 2;
|
|
|
|
BX_SER_THIS s[0].modem_cntl.out2 = (value & 0x08) >> 3;
|
|
|
|
BX_SER_THIS s[0].modem_cntl.local_loopback = (value & 0x10) >> 4;
|
|
|
|
|
|
|
|
if (BX_SER_THIS s[0].modem_cntl.local_loopback) {
|
2002-01-20 19:35:32 +03:00
|
|
|
prev_cts = BX_SER_THIS s[0].modem_status.cts;
|
|
|
|
prev_dsr = BX_SER_THIS s[0].modem_status.dsr;
|
|
|
|
prev_ri = BX_SER_THIS s[0].modem_status.ri;
|
|
|
|
prev_dcd = BX_SER_THIS s[0].modem_status.dcd;
|
|
|
|
BX_SER_THIS s[0].modem_status.cts = BX_SER_THIS s[0].modem_cntl.rts;
|
|
|
|
BX_SER_THIS s[0].modem_status.dsr = BX_SER_THIS s[0].modem_cntl.dtr;
|
|
|
|
BX_SER_THIS s[0].modem_status.ri = BX_SER_THIS s[0].modem_cntl.out1;
|
|
|
|
BX_SER_THIS s[0].modem_status.dcd = BX_SER_THIS s[0].modem_cntl.out2;
|
|
|
|
if (BX_SER_THIS s[0].modem_status.cts != prev_cts) {
|
|
|
|
BX_SER_THIS s[0].modem_status.delta_cts = 1;
|
|
|
|
BX_SER_THIS s[0].ms_ipending = 1;
|
|
|
|
}
|
|
|
|
if (BX_SER_THIS s[0].modem_status.dsr != prev_dsr) {
|
|
|
|
BX_SER_THIS s[0].modem_status.delta_dsr = 1;
|
|
|
|
BX_SER_THIS s[0].ms_ipending = 1;
|
|
|
|
}
|
|
|
|
if (BX_SER_THIS s[0].modem_status.ri != prev_ri)
|
|
|
|
BX_SER_THIS s[0].ms_ipending = 1;
|
|
|
|
if ((BX_SER_THIS s[0].modem_status.ri == 0) && (prev_ri == 1))
|
|
|
|
BX_SER_THIS s[0].modem_status.ri_trailedge = 1;
|
|
|
|
if (BX_SER_THIS s[0].modem_status.dcd != prev_dcd) {
|
|
|
|
BX_SER_THIS s[0].modem_status.delta_dcd = 1;
|
|
|
|
BX_SER_THIS s[0].ms_ipending = 1;
|
|
|
|
}
|
2003-10-31 20:23:56 +03:00
|
|
|
raise_interrupt(0, BX_SER_INT_MODSTAT);
|
2001-04-10 05:04:59 +04:00
|
|
|
} else {
|
2002-01-20 19:35:32 +03:00
|
|
|
/* set these to 0 for the time being */
|
|
|
|
BX_SER_THIS s[0].modem_status.cts = 0;
|
|
|
|
BX_SER_THIS s[0].modem_status.dsr = 0;
|
|
|
|
BX_SER_THIS s[0].modem_status.ri = 0;
|
|
|
|
BX_SER_THIS s[0].modem_status.dcd = 0;
|
2001-04-10 05:04:59 +04:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case 0x03FD: /* Line status register */
|
2001-11-17 21:10:54 +03:00
|
|
|
BX_ERROR(("write to line status register ignored"));
|
2001-04-10 05:04:59 +04:00
|
|
|
break;
|
|
|
|
|
|
|
|
case 0x03FE: /* MODEM status register */
|
2001-11-17 21:10:54 +03:00
|
|
|
BX_ERROR(("write to MODEM status register ignored"));
|
2001-04-10 05:04:59 +04:00
|
|
|
break;
|
|
|
|
|
|
|
|
case 0x03FF: /* scratch register */
|
|
|
|
BX_SER_THIS s[0].scratch = value;
|
|
|
|
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;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
bx_serial_c::tx_timer_handler(void *this_ptr)
|
|
|
|
{
|
|
|
|
bx_serial_c *class_ptr = (bx_serial_c *) this_ptr;
|
|
|
|
|
|
|
|
class_ptr->tx_timer();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
bx_serial_c::tx_timer(void)
|
|
|
|
{
|
|
|
|
if (BX_SER_THIS s[0].modem_cntl.local_loopback) {
|
2003-10-31 20:23:56 +03:00
|
|
|
if (BX_SER_THIS s[0].line_status.rxdata_ready == 1) {
|
2002-01-20 19:35:32 +03:00
|
|
|
BX_SER_THIS s[0].line_status.overrun_error = 1;
|
2003-10-31 20:23:56 +03:00
|
|
|
raise_interrupt(0, BX_SER_INT_RXLSTAT);
|
|
|
|
}
|
2003-09-15 00:16:25 +04:00
|
|
|
BX_SER_THIS s[0].rxbuffer = BX_SER_THIS s[0].tsrbuffer;
|
2001-04-10 05:04:59 +04:00
|
|
|
BX_SER_THIS s[0].line_status.rxdata_ready = 1;
|
2003-10-31 20:23:56 +03:00
|
|
|
raise_interrupt(0, BX_SER_INT_RXDATA);
|
2001-04-10 05:04:59 +04:00
|
|
|
} else {
|
2003-10-12 14:51:58 +04:00
|
|
|
#if USE_RAW_SERIAL
|
2001-04-10 05:04:59 +04:00
|
|
|
if (!BX_SER_THIS raw->ready_transmit())
|
2001-11-17 21:10:54 +03:00
|
|
|
BX_PANIC(("Not ready to transmit"));
|
2003-09-15 00:16:25 +04:00
|
|
|
BX_SER_THIS raw->transmit(BX_SER_THIS s[0].tsrbuffer);
|
2001-06-19 21:52:46 +04:00
|
|
|
#endif
|
|
|
|
#if defined(SERIAL_ENABLE)
|
2003-09-15 00:16:25 +04:00
|
|
|
BX_DEBUG(("write: '%c'", BX_SER_THIS s[0].tsrbuffer));
|
|
|
|
if (tty_id >= 0) write(tty_id, (bx_ptr_t) & BX_SER_THIS s[0].tsrbuffer, 1);
|
2001-04-10 05:04:59 +04:00
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
2003-09-15 00:16:25 +04:00
|
|
|
if (!BX_SER_THIS s[0].line_status.thr_empty) {
|
|
|
|
BX_SER_THIS s[0].tsrbuffer = BX_SER_THIS s[0].thrbuffer;
|
|
|
|
BX_SER_THIS s[0].line_status.thr_empty = 1;
|
2003-10-31 20:23:56 +03:00
|
|
|
raise_interrupt(0, BX_SER_INT_TXHOLD);
|
2003-09-15 00:16:25 +04:00
|
|
|
bx_pc_system.activate_timer(BX_SER_THIS s[0].tx_timer_index,
|
|
|
|
(int) (1000000.0 / BX_SER_THIS s[0].baudrate *
|
|
|
|
(BX_SER_THIS s[0].line_cntl.wordlen_sel + 5)),
|
|
|
|
0); /* not continuous */
|
|
|
|
} else {
|
|
|
|
BX_SER_THIS s[0].line_status.tsr_empty = 1;
|
|
|
|
}
|
2001-04-10 05:04:59 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
bx_serial_c::rx_timer_handler(void *this_ptr)
|
|
|
|
{
|
|
|
|
bx_serial_c *class_ptr = (bx_serial_c *) this_ptr;
|
|
|
|
|
|
|
|
class_ptr->rx_timer();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
bx_serial_c::rx_timer(void)
|
|
|
|
{
|
|
|
|
#if BX_HAVE_SELECT
|
2002-10-12 17:09:32 +04:00
|
|
|
#ifndef __BEOS__
|
2001-06-10 05:35:10 +04:00
|
|
|
struct timeval tval;
|
2001-04-10 05:04:59 +04:00
|
|
|
fd_set fds;
|
|
|
|
#endif
|
|
|
|
#endif
|
2003-10-12 14:51:58 +04:00
|
|
|
int bdrate = BX_SER_THIS s[0].baudrate / (BX_SER_THIS s[0].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
|
|
|
|
2001-06-10 05:35:10 +04:00
|
|
|
#if BX_HAVE_SELECT
|
2002-10-12 17:09:32 +04:00
|
|
|
#ifndef __BEOS__
|
2001-04-10 05:04:59 +04:00
|
|
|
tval.tv_sec = 0;
|
|
|
|
tval.tv_usec = 0;
|
|
|
|
|
|
|
|
// 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.
|
|
|
|
|
2002-03-03 09:03:29 +03:00
|
|
|
FD_ZERO(&fds);
|
2002-08-24 23:03:43 +04:00
|
|
|
if (tty_id >= 0) FD_SET(tty_id, &fds);
|
2001-04-10 05:04:59 +04:00
|
|
|
|
|
|
|
if (BX_SER_THIS s[0].line_status.rxdata_ready == 0) {
|
2003-10-12 14:51:58 +04:00
|
|
|
#if USE_RAW_SERIAL
|
2002-10-25 15:44:41 +04:00
|
|
|
bx_bool rdy;
|
2001-04-10 05:04:59 +04:00
|
|
|
uint16 data;
|
|
|
|
if ((rdy = BX_SER_THIS raw->ready_receive())) {
|
|
|
|
data = BX_SER_THIS raw->receive();
|
|
|
|
if (data == C_BREAK) {
|
2001-11-17 21:10:54 +03:00
|
|
|
BX_DEBUG(("got BREAK"));
|
2001-04-10 05:04:59 +04:00
|
|
|
BX_SER_THIS s[0].line_status.break_int = 1;
|
|
|
|
rdy = 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (rdy) {
|
2003-10-12 14:51:58 +04:00
|
|
|
chbuf = data;
|
2001-06-19 21:52:46 +04:00
|
|
|
#elif defined(SERIAL_ENABLE)
|
2002-08-24 23:03:43 +04:00
|
|
|
if ((tty_id >= 0) && (select(tty_id + 1, &fds, NULL, NULL, &tval) == 1)) {
|
2001-06-19 21:52:46 +04:00
|
|
|
(void) read(tty_id, &chbuf, 1);
|
2003-10-12 14:51:58 +04:00
|
|
|
BX_DEBUG(("read: '%c'",chbuf));
|
2001-04-10 05:04:59 +04:00
|
|
|
#else
|
|
|
|
if (0) {
|
|
|
|
#endif
|
|
|
|
if (!BX_SER_THIS s[0].modem_cntl.local_loopback) {
|
2003-10-31 20:23:56 +03:00
|
|
|
if (BX_SER_THIS s[0].line_status.rxdata_ready == 1) {
|
|
|
|
BX_SER_THIS s[0].line_status.overrun_error = 1;
|
|
|
|
raise_interrupt(0, BX_SER_INT_RXLSTAT);
|
|
|
|
}
|
2002-01-20 19:35:32 +03:00
|
|
|
BX_SER_THIS s[0].rxbuffer = chbuf;
|
|
|
|
BX_SER_THIS s[0].line_status.rxdata_ready = 1;
|
2003-10-31 20:23:56 +03:00
|
|
|
raise_interrupt(0, BX_SER_INT_RXDATA);
|
2001-04-10 05:04:59 +04:00
|
|
|
}
|
|
|
|
} else {
|
2002-03-05 18:40:23 +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
|
|
|
}
|
|
|
|
#endif
|
|
|
|
#endif
|
|
|
|
|
|
|
|
bx_pc_system.activate_timer(BX_SER_THIS s[0].rx_timer_index,
|
|
|
|
(int) (1000000.0 / bdrate),
|
|
|
|
0); /* not continuous */
|
|
|
|
}
|