2001-10-03 17:10:38 +04:00
|
|
|
/////////////////////////////////////////////////////////////////////////
|
Now, when you compile with --enable-guest2host-tlb, non-paged
mode uses the notion of the guest-to-host TLB. This has the
benefit of allowing more uniform and streamlined acceleration
code in access.cc which does not have to check if CR0.PG
is set, eliminating a few instructions per guest access.
Shaved just a little off execution time, as expected.
Also, access_linear now breaks accesses which span two pages,
into two calls the the physical memory routines, when paging
is off, just like it always has for paging on. Besides
being more uniform, this allows the physical memory access
routines to known the complete data item is contained
within a single physical page, and stop reapplying the
A20ADDR() macro to pointers as it increments them.
Perhaps things can be optimized a little more now there too...
I renamed the routines to {read,write}PhysicalPage() as
a reminder that these routines now operate on data
solely within one page.
I also added a little code so that the paging module is
notified when the A20 line is tweaked, so it can dump
whatever mappings it wants to.
2002-09-05 06:31:24 +04:00
|
|
|
// $Id: pc_system.cc,v 1.22 2002-09-05 02:31:23 kevinlawton Exp $
|
2001-10-03 17:10:38 +04:00
|
|
|
/////////////////////////////////////////////////////////////////////////
|
|
|
|
//
|
2002-06-16 19:02:28 +04: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
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
#include "bochs.h"
|
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
|
|
|
#define LOG_THIS bx_pc_system.
|
2001-04-10 05:04:59 +04:00
|
|
|
|
|
|
|
#ifdef WIN32
|
|
|
|
#ifndef __MINGW32__
|
|
|
|
// #include <winsock2.h> // +++
|
|
|
|
#include <winsock.h>
|
|
|
|
#endif
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#if BX_SHOW_IPS
|
|
|
|
unsigned long ips_count=0;
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#if defined(PROVIDE_M_IPS)
|
|
|
|
double m_ips; // Millions of Instructions Per Second
|
|
|
|
#endif
|
|
|
|
|
|
|
|
const Bit64u bx_pc_system_c::COUNTER_INTERVAL = 100000;
|
|
|
|
|
|
|
|
// constructor
|
|
|
|
bx_pc_system_c::bx_pc_system_c(void)
|
|
|
|
{
|
2001-06-27 23:16:01 +04:00
|
|
|
this->put("SYS");
|
2002-08-27 22:53:30 +04:00
|
|
|
}
|
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
|
|
|
|
2002-08-27 22:53:30 +04:00
|
|
|
void
|
|
|
|
bx_pc_system_c::init_ips(Bit32u ips)
|
|
|
|
{
|
2001-04-10 05:04:59 +04:00
|
|
|
num_timers = 0;
|
|
|
|
// set ticks period and remaining to max Bit32u value
|
|
|
|
num_cpu_ticks_in_period = num_cpu_ticks_left = (Bit32u) -1;
|
|
|
|
m_ips = 0.0L;
|
|
|
|
|
|
|
|
HRQ = 0;
|
|
|
|
|
|
|
|
enable_a20 = 1;
|
2001-05-23 12:16:07 +04:00
|
|
|
//set_INTR (0);
|
2001-04-10 05:04:59 +04:00
|
|
|
|
|
|
|
#if BX_CPU_LEVEL < 2
|
|
|
|
a20_mask = 0xfffff;
|
|
|
|
#elif BX_CPU_LEVEL == 2
|
|
|
|
a20_mask = 0xffffff;
|
|
|
|
#else /* 386+ */
|
|
|
|
a20_mask = 0xffffffff;
|
|
|
|
#endif
|
|
|
|
|
|
|
|
counter = 0;
|
|
|
|
counter_timer_index = register_timer_ticks(this, bx_pc_system_c::counter_timer_handler, COUNTER_INTERVAL, 1, 1);
|
|
|
|
|
|
|
|
// parameter 'ips' is the processor speed in Instructions-Per-Second
|
|
|
|
m_ips = double(ips) / 1000000.0L;
|
2001-05-30 22:56:02 +04:00
|
|
|
BX_DEBUG(("ips = %u", (unsigned) ips));
|
2001-04-10 05:04:59 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
bx_pc_system_c::set_HRQ(Boolean val)
|
|
|
|
{
|
|
|
|
HRQ = val;
|
|
|
|
if (val)
|
2001-05-23 12:16:07 +04:00
|
|
|
BX_CPU(0)->async_event = 1;
|
2001-12-18 16:14:46 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2001-04-10 05:04:59 +04:00
|
|
|
#if (BX_NUM_SIMULATORS < 2)
|
|
|
|
void
|
|
|
|
bx_pc_system_c::set_INTR(Boolean value)
|
|
|
|
{
|
2001-05-23 12:16:07 +04:00
|
|
|
if (bx_dbg.interrupts)
|
2001-05-30 22:56:02 +04:00
|
|
|
BX_INFO(("pc_system: Setting INTR=%d on bootstrap processor %d", (int)value, BX_BOOTSTRAP_PROCESSOR));
|
2001-05-23 12:16:07 +04:00
|
|
|
//INTR = value;
|
|
|
|
int cpu = BX_BOOTSTRAP_PROCESSOR;
|
|
|
|
BX_CPU(cpu)->set_INTR(value);
|
2001-04-10 05:04:59 +04:00
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
//
|
|
|
|
// Read from the IO memory address space
|
|
|
|
//
|
|
|
|
|
|
|
|
Bit32u
|
|
|
|
bx_pc_system_c::inp(Bit16u addr, unsigned io_len)
|
|
|
|
{
|
|
|
|
Bit32u ret;
|
|
|
|
|
|
|
|
ret = bx_devices.inp(addr, io_len);
|
|
|
|
|
|
|
|
return( ret );
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
//
|
|
|
|
// Write to the IO memory address space.
|
|
|
|
//
|
|
|
|
|
|
|
|
void
|
|
|
|
bx_pc_system_c::outp(Bit16u addr, Bit32u value, unsigned io_len)
|
|
|
|
{
|
|
|
|
bx_devices.outp(addr, value, io_len);
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
bx_pc_system_c::set_enable_a20(Bit8u value)
|
|
|
|
{
|
|
|
|
#if BX_CPU_LEVEL < 2
|
2001-05-30 22:56:02 +04:00
|
|
|
BX_PANIC(("set_enable_a20() called: 8086 emulation"));
|
2001-04-10 05:04:59 +04:00
|
|
|
#else
|
|
|
|
|
|
|
|
#if BX_SUPPORT_A20
|
Now, when you compile with --enable-guest2host-tlb, non-paged
mode uses the notion of the guest-to-host TLB. This has the
benefit of allowing more uniform and streamlined acceleration
code in access.cc which does not have to check if CR0.PG
is set, eliminating a few instructions per guest access.
Shaved just a little off execution time, as expected.
Also, access_linear now breaks accesses which span two pages,
into two calls the the physical memory routines, when paging
is off, just like it always has for paging on. Besides
being more uniform, this allows the physical memory access
routines to known the complete data item is contained
within a single physical page, and stop reapplying the
A20ADDR() macro to pointers as it increments them.
Perhaps things can be optimized a little more now there too...
I renamed the routines to {read,write}PhysicalPage() as
a reminder that these routines now operate on data
solely within one page.
I also added a little code so that the paging module is
notified when the A20 line is tweaked, so it can dump
whatever mappings it wants to.
2002-09-05 06:31:24 +04:00
|
|
|
unsigned old_enable_a20 = enable_a20;
|
|
|
|
|
2001-04-10 05:04:59 +04:00
|
|
|
if (value) {
|
|
|
|
enable_a20 = 1;
|
|
|
|
#if BX_CPU_LEVEL == 2
|
|
|
|
a20_mask = 0xffffff; /* 286: enable all 24 address lines */
|
|
|
|
#else /* 386+ */
|
|
|
|
a20_mask = 0xffffffff; /* 386: enable all 32 address lines */
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
enable_a20 = 0;
|
|
|
|
a20_mask = 0xffefffff; /* mask off A20 address line */
|
|
|
|
}
|
|
|
|
|
|
|
|
BX_DBG_A20_REPORT(value);
|
|
|
|
|
2001-05-30 22:56:02 +04:00
|
|
|
BX_DEBUG(("A20: set() = %u", (unsigned) enable_a20));
|
Now, when you compile with --enable-guest2host-tlb, non-paged
mode uses the notion of the guest-to-host TLB. This has the
benefit of allowing more uniform and streamlined acceleration
code in access.cc which does not have to check if CR0.PG
is set, eliminating a few instructions per guest access.
Shaved just a little off execution time, as expected.
Also, access_linear now breaks accesses which span two pages,
into two calls the the physical memory routines, when paging
is off, just like it always has for paging on. Besides
being more uniform, this allows the physical memory access
routines to known the complete data item is contained
within a single physical page, and stop reapplying the
A20ADDR() macro to pointers as it increments them.
Perhaps things can be optimized a little more now there too...
I renamed the routines to {read,write}PhysicalPage() as
a reminder that these routines now operate on data
solely within one page.
I also added a little code so that the paging module is
notified when the A20 line is tweaked, so it can dump
whatever mappings it wants to.
2002-09-05 06:31:24 +04:00
|
|
|
|
|
|
|
// If there has been a transition, we need to notify the CPUs so
|
|
|
|
// they can potentially invalidate certain cache info based on
|
|
|
|
// A20-line-applied physical addresses.
|
|
|
|
if (old_enable_a20 != enable_a20) {
|
|
|
|
for (unsigned i=0; i<BX_SMP_PROCESSORS; i++)
|
|
|
|
BX_CPU(i)->pagingA20Changed();
|
|
|
|
}
|
2001-04-10 05:04:59 +04:00
|
|
|
#else
|
2001-05-30 22:56:02 +04:00
|
|
|
BX_DEBUG(("set_enable_a20: ignoring: SUPPORT_A20 = 0"));
|
2001-04-10 05:04:59 +04:00
|
|
|
#endif // #if BX_SUPPORT_A20
|
|
|
|
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
Boolean
|
|
|
|
bx_pc_system_c::get_enable_a20(void)
|
|
|
|
{
|
|
|
|
#if BX_SUPPORT_A20
|
|
|
|
if (bx_dbg.a20)
|
2001-05-30 22:56:02 +04:00
|
|
|
BX_INFO(("A20: get() = %u", (unsigned) enable_a20));
|
2001-04-10 05:04:59 +04:00
|
|
|
|
|
|
|
if (enable_a20) return(1);
|
|
|
|
else return(0);
|
|
|
|
#else
|
2001-05-30 22:56:02 +04:00
|
|
|
BX_INFO(("get_enable_a20: ignoring: SUPPORT_A20 = 0"));
|
2001-04-10 05:04:59 +04:00
|
|
|
return(1);
|
|
|
|
#endif // #if BX_SUPPORT_A20
|
|
|
|
}
|
|
|
|
|
|
|
|
int
|
|
|
|
bx_pc_system_c::ResetSignal( PCS_OP operation )
|
|
|
|
{
|
|
|
|
UNUSED( operation );
|
|
|
|
// Reset the processor.
|
|
|
|
|
2001-05-30 22:56:02 +04:00
|
|
|
BX_ERROR(( "# bx_pc_system_c::ResetSignal() called" ));
|
2001-05-23 12:16:07 +04:00
|
|
|
for (int i=0; i<BX_SMP_PROCESSORS; i++)
|
|
|
|
BX_CPU(i)->reset(BX_RESET_SOFTWARE);
|
2002-08-28 01:30:48 +04:00
|
|
|
bx_devices.reset(BX_RESET_SOFTWARE);
|
2001-04-10 05:04:59 +04:00
|
|
|
return(0);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
Bit8u
|
|
|
|
bx_pc_system_c::IAC(void)
|
|
|
|
{
|
|
|
|
return( bx_devices.pic->IAC() );
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
bx_pc_system_c::exit(void)
|
|
|
|
{
|
|
|
|
if (bx_devices.hard_drive)
|
|
|
|
bx_devices.hard_drive->close_harddrive();
|
2001-05-30 22:56:02 +04:00
|
|
|
BX_INFO(("Last time is %d", bx_cmos.s.timeval));
|
2001-04-10 05:04:59 +04:00
|
|
|
bx_gui.exit();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
//
|
|
|
|
// bochs timer support
|
|
|
|
//
|
|
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
bx_pc_system_c::timer_handler(void)
|
|
|
|
{
|
|
|
|
Bit64u min;
|
|
|
|
unsigned i;
|
|
|
|
Bit64u delta;
|
|
|
|
|
2001-05-30 22:56:02 +04:00
|
|
|
// BX_ERROR(( "Time handler ptime = %d", bx_pc_system.time_ticks() ));
|
2001-04-10 05:04:59 +04:00
|
|
|
|
|
|
|
delta = num_cpu_ticks_in_period - num_cpu_ticks_left;
|
|
|
|
#if BX_TIMER_DEBUG
|
|
|
|
if (num_cpu_ticks_left != 0)
|
2001-05-30 22:56:02 +04:00
|
|
|
BX_PANIC(("timer_handler: ticks_left!=0"));
|
2001-04-10 05:04:59 +04:00
|
|
|
#endif
|
|
|
|
|
|
|
|
for (i=0; i < num_timers; i++) {
|
|
|
|
timer[i].triggered = 0;
|
|
|
|
if (timer[i].active) {
|
|
|
|
#if BX_TIMER_DEBUG
|
|
|
|
if (timer[i].remaining < delta) {
|
2001-05-30 22:56:02 +04:00
|
|
|
BX_PANIC(("timer_handler: remain < delta"));
|
2001-04-10 05:04:59 +04:00
|
|
|
}
|
|
|
|
#endif
|
|
|
|
timer[i].remaining -= delta;
|
|
|
|
if (timer[i].remaining == 0) {
|
|
|
|
timer[i].triggered = 1;
|
|
|
|
// reset remaining period for triggered timer
|
|
|
|
timer[i].remaining = timer[i].period;
|
|
|
|
|
|
|
|
// if triggered timer is one-shot, deactive
|
|
|
|
if (timer[i].continuous==0)
|
|
|
|
timer[i].active = 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
min = (Bit64u) -1; // max number in Bit64u range
|
|
|
|
for (i=0; i < num_timers; i++) {
|
|
|
|
if (timer[i].active && (timer[i].remaining < min))
|
|
|
|
min = timer[i].remaining;
|
|
|
|
}
|
|
|
|
num_cpu_ticks_in_period = num_cpu_ticks_left = min;
|
|
|
|
|
|
|
|
for (i=0; i < num_timers; i++) {
|
|
|
|
// call requested timer function. It may request a different
|
|
|
|
// timer period or deactivate, all cases handled below
|
|
|
|
if (timer[i].triggered) {
|
|
|
|
timer[i].funct(timer[i].this_ptr);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
bx_pc_system_c::expire_ticks(void)
|
|
|
|
{
|
|
|
|
unsigned i;
|
|
|
|
Bit64u ticks_delta;
|
|
|
|
|
|
|
|
ticks_delta = num_cpu_ticks_in_period - num_cpu_ticks_left;
|
|
|
|
if (ticks_delta == 0) return; // no ticks occurred since
|
|
|
|
for (i=0; i<num_timers; i++) {
|
|
|
|
if (timer[i].active) {
|
|
|
|
#if BX_TIMER_DEBUG
|
|
|
|
if (timer[i].remaining <= ticks_delta) {
|
|
|
|
for (unsigned j=0; j<num_timers; j++) {
|
2001-05-30 22:56:02 +04:00
|
|
|
BX_INFO(("^^^timer[%u]", j));
|
|
|
|
BX_INFO(("^^^remaining = %u, period = %u",
|
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
|
|
|
timer[j].remaining, timer[j].period));
|
2001-04-10 05:04:59 +04:00
|
|
|
}
|
2001-05-30 22:56:02 +04:00
|
|
|
BX_PANIC(("expire_ticks: i=%u, remain(%u) <= delta(%u)",
|
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
|
|
|
i, timer[i].remaining, (unsigned) ticks_delta));
|
2001-04-10 05:04:59 +04:00
|
|
|
}
|
|
|
|
#endif
|
|
|
|
timer[i].remaining -= ticks_delta; // must be >= 1 here
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// set new period to number of ticks left
|
|
|
|
num_cpu_ticks_in_period = num_cpu_ticks_left;
|
|
|
|
}
|
|
|
|
|
|
|
|
int
|
|
|
|
bx_pc_system_c::register_timer( void *this_ptr, void (*funct)(void *),
|
|
|
|
Bit32u useconds, Boolean continuous, Boolean active)
|
|
|
|
{
|
|
|
|
Bit64u instructions;
|
|
|
|
|
|
|
|
if (num_timers >= BX_MAX_TIMERS) {
|
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
|
|
|
BX_PANIC(("register_timer: too many registered timers."));
|
2001-04-10 05:04:59 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
if (this_ptr == NULL)
|
2001-05-30 22:56:02 +04:00
|
|
|
BX_PANIC(("register_timer: this_ptr is NULL"));
|
2001-04-10 05:04:59 +04:00
|
|
|
if (funct == NULL)
|
2001-05-30 22:56:02 +04:00
|
|
|
BX_PANIC(("register_timer: funct is NULL"));
|
2001-04-10 05:04:59 +04:00
|
|
|
|
|
|
|
// account for ticks up to now
|
|
|
|
expire_ticks();
|
|
|
|
|
|
|
|
// convert useconds to number of instructions
|
|
|
|
instructions = (Bit64u) (double(useconds) * m_ips);
|
2001-08-18 07:28:23 +04:00
|
|
|
if((useconds!=0) && (instructions==0)) instructions = 1;
|
2001-04-10 05:04:59 +04:00
|
|
|
|
|
|
|
return register_timer_ticks(this_ptr, funct, instructions, continuous, active);
|
|
|
|
}
|
|
|
|
|
|
|
|
int
|
|
|
|
bx_pc_system_c::register_timer_ticks(void* this_ptr, bx_timer_handler_t funct, Bit64u instructions, Boolean continuous, Boolean active)
|
|
|
|
{
|
|
|
|
unsigned i;
|
|
|
|
|
|
|
|
if (num_timers >= BX_MAX_TIMERS) {
|
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
|
|
|
BX_PANIC(("register_timer: too many registered timers."));
|
2001-04-10 05:04:59 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
if (this_ptr == NULL)
|
2001-05-30 22:56:02 +04:00
|
|
|
BX_PANIC(("register_timer: this_ptr is NULL"));
|
2001-04-10 05:04:59 +04:00
|
|
|
if (funct == NULL)
|
2001-05-30 22:56:02 +04:00
|
|
|
BX_PANIC(("register_timer: funct is NULL"));
|
2001-04-10 05:04:59 +04:00
|
|
|
|
|
|
|
i = num_timers;
|
|
|
|
num_timers++;
|
|
|
|
timer[i].period = instructions;
|
|
|
|
timer[i].remaining = instructions;
|
|
|
|
timer[i].active = active;
|
|
|
|
timer[i].funct = funct;
|
|
|
|
timer[i].continuous = continuous;
|
|
|
|
timer[i].this_ptr = this_ptr;
|
|
|
|
|
|
|
|
if (active) {
|
|
|
|
if (num_cpu_ticks_in_period == 0) {
|
|
|
|
// no active timers
|
|
|
|
num_cpu_ticks_in_period = instructions;
|
|
|
|
num_cpu_ticks_left = instructions;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
if (instructions < num_cpu_ticks_left) {
|
|
|
|
num_cpu_ticks_in_period = instructions;
|
|
|
|
num_cpu_ticks_left = instructions;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// return timer id
|
|
|
|
return(i);
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
bx_pc_system_c::counter_timer_handler(void* this_ptr)
|
|
|
|
{
|
|
|
|
UNUSED(this_ptr);
|
|
|
|
|
|
|
|
bx_pc_system.counter++;
|
|
|
|
}
|
|
|
|
|
|
|
|
#if BX_DEBUGGER
|
|
|
|
void
|
|
|
|
bx_pc_system_c::timebp_handler(void* this_ptr)
|
|
|
|
{
|
2001-05-23 12:16:07 +04:00
|
|
|
BX_CPU(0)->break_point = BREAK_POINT_TIME;
|
2001-05-30 22:56:02 +04:00
|
|
|
BX_DEBUG(( "Time breakpoint triggered" ));
|
2001-04-10 05:04:59 +04:00
|
|
|
|
|
|
|
if (timebp_queue_size > 1) {
|
2001-09-24 04:35:42 +04:00
|
|
|
Bit64s new_diff = timebp_queue[1] - bx_pc_system.time_ticks();
|
2001-04-10 05:04:59 +04:00
|
|
|
bx_pc_system.activate_timer_ticks(timebp_timer, new_diff, 1);
|
|
|
|
}
|
|
|
|
timebp_queue_size--;
|
|
|
|
for (int i = 0; i < timebp_queue_size; i++)
|
|
|
|
timebp_queue[i] = timebp_queue[i+1];
|
|
|
|
}
|
|
|
|
#endif // BX_DEBUGGER
|
|
|
|
|
2001-07-02 00:42:56 +04:00
|
|
|
Bit64u
|
|
|
|
bx_pc_system_c::time_usec() {
|
2001-08-14 10:24:12 +04:00
|
|
|
return (Bit64u) (((double)(Bit64s)time_ticks()) / m_ips );
|
2001-07-02 00:42:56 +04:00
|
|
|
}
|
2001-04-10 05:04:59 +04:00
|
|
|
|
|
|
|
Bit64u
|
|
|
|
bx_pc_system_c::time_ticks()
|
|
|
|
{
|
|
|
|
return (counter + 1) * COUNTER_INTERVAL
|
|
|
|
- ticks_remaining(counter_timer_index)
|
|
|
|
+ ((Bit64u)num_cpu_ticks_in_period - (Bit64u)num_cpu_ticks_left);
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
bx_pc_system_c::start_timers(void)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
bx_pc_system_c::activate_timer_ticks (unsigned timer_index, Bit64u instructions, Boolean continuous)
|
|
|
|
{
|
|
|
|
if (timer_index >= num_timers)
|
2001-05-30 22:56:02 +04:00
|
|
|
BX_PANIC(("activate_timer(): bad timer index given"));
|
2001-04-10 05:04:59 +04:00
|
|
|
|
|
|
|
// set timer continuity to new value (1=continuous, 0=one-shot)
|
|
|
|
timer[timer_index].continuous = continuous;
|
|
|
|
|
|
|
|
timer[timer_index].active = 1;
|
|
|
|
timer[timer_index].remaining = instructions;
|
|
|
|
|
|
|
|
if (num_cpu_ticks_in_period == 0) {
|
|
|
|
// no active timers
|
|
|
|
num_cpu_ticks_in_period = instructions;
|
|
|
|
num_cpu_ticks_left = instructions;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
if (instructions < num_cpu_ticks_left) {
|
|
|
|
num_cpu_ticks_in_period = instructions;
|
|
|
|
num_cpu_ticks_left = instructions;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
bx_pc_system_c::activate_timer( unsigned timer_index,
|
|
|
|
Bit32u useconds, Boolean continuous )
|
|
|
|
{
|
|
|
|
Bit64u instructions;
|
|
|
|
|
|
|
|
if (timer_index >= num_timers)
|
2001-05-30 22:56:02 +04:00
|
|
|
BX_PANIC(("activate_timer(): bad timer index given"));
|
2001-04-10 05:04:59 +04:00
|
|
|
|
|
|
|
// account for ticks up to now
|
|
|
|
expire_ticks();
|
|
|
|
|
|
|
|
// set timer continuity to new value (1=continuous, 0=one-shot)
|
|
|
|
timer[timer_index].continuous = continuous;
|
|
|
|
|
|
|
|
// if useconds = 0, use default stored in period field
|
|
|
|
// else set new period from useconds
|
|
|
|
if (useconds==0)
|
|
|
|
instructions = timer[timer_index].period;
|
|
|
|
else {
|
|
|
|
// convert useconds to number of instructions
|
|
|
|
instructions = (Bit64u) (double(useconds) * m_ips);
|
2001-08-18 07:28:23 +04:00
|
|
|
if(instructions==0) instructions = 1;
|
2001-04-10 05:04:59 +04:00
|
|
|
timer[timer_index].period = instructions;
|
|
|
|
}
|
|
|
|
|
|
|
|
timer[timer_index].active = 1;
|
|
|
|
timer[timer_index].remaining = instructions;
|
|
|
|
|
|
|
|
if (num_cpu_ticks_in_period == 0) {
|
|
|
|
// no active timers
|
|
|
|
num_cpu_ticks_in_period = instructions;
|
|
|
|
num_cpu_ticks_left = instructions;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
if (instructions < num_cpu_ticks_left) {
|
|
|
|
num_cpu_ticks_in_period = instructions;
|
|
|
|
num_cpu_ticks_left = instructions;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
bx_pc_system_c::deactivate_timer( unsigned timer_index )
|
|
|
|
{
|
|
|
|
if (timer_index >= num_timers)
|
2001-05-30 22:56:02 +04:00
|
|
|
BX_PANIC(("deactivate_timer(): bad timer index given"));
|
2001-04-10 05:04:59 +04:00
|
|
|
|
|
|
|
timer[timer_index].active = 0;
|
|
|
|
}
|