Bochs/bochs/iodev/guest2host.cc
2001-04-10 01:04:59 +00:00

143 lines
3.6 KiB
C++

// Copyright (C) 2000 MandrakeSoft S.A.
//
// 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"
bx_g2h_c bx_g2h;
bx_g2h_c::bx_g2h_c(void)
{
unsigned i;
for (i=0; i<BX_MAX_G2H_CHANNELS; i++) {
s.callback[i].f = NULL;
s.callback[i].used = 0;
}
}
bx_g2h_c::~bx_g2h_c(void)
{
// nothing for now
}
void
bx_g2h_c::init(bx_devices_c *d)
{
// Reserve a dword port for this interface
for (Bit32u addr=BX_G2H_PORT; addr<=(BX_G2H_PORT+3); addr++) {
d->register_io_read_handler(&bx_g2h,
inp_handler, addr, "g2h");
d->register_io_write_handler(&bx_g2h,
outp_handler, addr, "g2h");
}
memset(&bx_g2h.s, 0, sizeof(bx_g2h.s));
}
unsigned
bx_g2h_c::aquire_channel(bx_g2h_callback_t f)
{
unsigned i;
for (i=0; i<BX_MAX_G2H_CHANNELS; i++) {
if (bx_g2h.s.callback[i].used==0) {
bx_g2h.s.callback[i].f = f;
bx_g2h.s.callback[i].used = 1;
return(i);
}
}
bx_printf("g2h: attempt to aquire channel: maxed out\n");
return(BX_G2H_ERROR); // No more free channels
}
unsigned
bx_g2h_c::deaquire_channel(unsigned channel)
{
if ( (channel >= BX_MAX_G2H_CHANNELS) ||
(bx_g2h.s.callback[channel].used==0) ) {
bx_panic("g2h: attempt to deaquire channel %u: not aquired\n",
channel);
}
bx_g2h.s.callback[channel].used = 0;
bx_g2h.s.callback[channel].f = NULL;
return(0);
}
// static IO port read callback handler
// redirects to non-static class handler to avoid virtual functions
Bit32u
bx_g2h_c::inp_handler(void *this_ptr, Bit32u addr, unsigned io_len)
{
UNUSED(this_ptr);
if (addr != BX_G2H_PORT)
bx_panic("g2h: IO read not aligned on dword boundary.\n");
if (io_len != 4)
bx_panic("g2h: IO read not dword.\n");
bx_panic("g2h: IO read not complete.\n");
return(0);
}
void
bx_g2h_c::outp_handler(void *this_ptr, Bit32u addr,
Bit32u val32, unsigned io_len)
{
UNUSED(this_ptr);
if (addr != BX_G2H_PORT)
bx_panic("g2h: IO write not aligned on dword boundary.\n");
if (io_len != 4)
bx_panic("g2h: IO write not dword.\n");
if ( (bx_g2h.s.packet_count==0) && (val32!=BX_G2H_MAGIC) ) {
bx_printf("g2h: IO W: Not magic header.\n");
return;
}
bx_g2h.s.guest_packet[bx_g2h.s.packet_count++] = val32;
if (bx_g2h.s.packet_count >= BX_G2H_PACKET_SIZE) {
unsigned channel;
// Full packet received from guest. Pass on to the host code.
channel = bx_g2h.s.guest_packet[1];
if (channel >= BX_MAX_G2H_CHANNELS) {
bx_panic("g2h: channel (%u) out of bounds\n", channel);
}
if (bx_g2h.s.callback[channel].used==0) {
bx_panic("g2h: channel (%u) not active\n", channel);
}
bx_g2h.s.callback[channel].f(&bx_g2h.s.guest_packet);
bx_g2h.s.packet_count = 0; // Ready for next packet
}
}