937a4bdc12
for HW devices.
230 lines
4.8 KiB
C
230 lines
4.8 KiB
C
/* This file is part of the program psim.
|
||
|
||
Copyright (C) 1994-1998, Andrew Cagney <cagney@highland.com.au>
|
||
|
||
This program is free software; you can redistribute it and/or modify
|
||
it under the terms of the GNU General Public License as published by
|
||
the Free Software Foundation; either version 2 of the License, or
|
||
(at your option) any later version.
|
||
|
||
This program 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 General Public License for more details.
|
||
|
||
You should have received a copy of the GNU General Public License
|
||
along with this program; if not, write to the Free Software
|
||
Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
|
||
|
||
*/
|
||
|
||
|
||
#include "sim-main.h"
|
||
|
||
#include "hw-device.h"
|
||
#include "hw-properties.h"
|
||
|
||
#if HAVE_STDLIB_H
|
||
#include <stdlib.h>
|
||
#endif
|
||
|
||
|
||
/* Address methods */
|
||
|
||
const hw_unit *
|
||
hw_unit_address (struct hw *me)
|
||
{
|
||
return &me->unit_address_of_hw;
|
||
}
|
||
|
||
|
||
|
||
/* IOCTL: */
|
||
|
||
int
|
||
hw_ioctl (struct hw *me,
|
||
sim_cpu *processor,
|
||
sim_cia cia,
|
||
hw_ioctl_request request,
|
||
...)
|
||
{
|
||
int status;
|
||
va_list ap;
|
||
va_start(ap, request);
|
||
status = me->to_ioctl (me, processor, cia, request, ap);
|
||
va_end(ap);
|
||
return status;
|
||
}
|
||
|
||
/* I/O */
|
||
|
||
void volatile
|
||
hw_abort (struct hw *me,
|
||
const char *fmt,
|
||
...)
|
||
{
|
||
SIM_DESC sd;
|
||
const char *name;
|
||
va_list ap;
|
||
va_start(ap, fmt);
|
||
/* find a system to abort through */
|
||
if (me == NULL || hw_system (me) == NULL)
|
||
sd = NULL;
|
||
else
|
||
sd = hw_system (me);
|
||
/* find an identity */
|
||
if (me != NULL && hw_path (me) != NULL && hw_path (me) [0] != '\0')
|
||
name = hw_path (me);
|
||
else if (me != NULL && hw_name (me) != NULL && hw_name (me)[0] != '\0')
|
||
name = hw_name (me);
|
||
else if (me != NULL && hw_family (me) != NULL && hw_family (me)[0] != '\0')
|
||
name = hw_family (me);
|
||
else
|
||
name = "device";
|
||
/* report the problem */
|
||
sim_io_eprintf (sd, "%s: ", name);
|
||
sim_io_evprintf (sd, fmt, ap);
|
||
sim_io_error (sd, "%s", "");
|
||
}
|
||
|
||
void
|
||
hw_trace (struct hw *me,
|
||
const char *fmt,
|
||
...)
|
||
{
|
||
if (hw_trace_p (me)) /* to be sure, to be sure */
|
||
{
|
||
va_list ap;
|
||
va_start (ap, fmt);
|
||
sim_io_eprintf (hw_system (me), "%s: ", hw_path (me));
|
||
sim_io_evprintf (hw_system (me), fmt, ap);
|
||
sim_io_eprintf (hw_system (me), "\n");
|
||
va_end (ap);
|
||
}
|
||
}
|
||
|
||
|
||
/* The event queue abstraction (for devices) */
|
||
|
||
|
||
struct _hw_event {
|
||
void *data;
|
||
struct hw *me;
|
||
hw_event_handler *handler;
|
||
sim_event *real;
|
||
};
|
||
|
||
/* Pass the H/W event onto the real handler */
|
||
|
||
static void
|
||
bounce_hw_event (SIM_DESC sd,
|
||
void *data)
|
||
{
|
||
hw_event event = * (hw_event*) data;
|
||
zfree (data);
|
||
event.handler (event.me, event.data);
|
||
}
|
||
|
||
|
||
/* Map onto the event functions */
|
||
|
||
hw_event *
|
||
hw_event_queue_schedule (struct hw *me,
|
||
signed64 delta_time,
|
||
hw_event_handler *handler,
|
||
void *data)
|
||
{
|
||
hw_event *event = ZALLOC (hw_event);
|
||
event->data = data;
|
||
event->handler = handler;
|
||
event->me = me;
|
||
event->real = sim_events_schedule (hw_system (me),
|
||
delta_time,
|
||
bounce_hw_event,
|
||
event);
|
||
return event;
|
||
}
|
||
|
||
void
|
||
hw_event_queue_deschedule (struct hw *me,
|
||
hw_event *event_to_remove)
|
||
{
|
||
sim_events_deschedule (hw_system (me),
|
||
event_to_remove->real);
|
||
zfree (event_to_remove);
|
||
}
|
||
|
||
signed64
|
||
hw_event_queue_time (struct hw *me)
|
||
{
|
||
return sim_events_time (hw_system (me));
|
||
}
|
||
|
||
|
||
/* Mechanism for associating allocated memory regions to a device.
|
||
When a device is deleted any remaining memory regions are also
|
||
reclaimed.
|
||
|
||
FIXME: Perhaphs this can be generalized, perhaphs it should not
|
||
be. */
|
||
|
||
struct hw_alloc_data {
|
||
void *alloc;
|
||
int zalloc_p;
|
||
struct hw_alloc_data *next;
|
||
};
|
||
|
||
extern void *
|
||
hw_zalloc (struct hw *me, unsigned long size)
|
||
{
|
||
struct hw_alloc_data *memory = ZALLOC (struct hw_alloc_data);
|
||
memory->alloc = zalloc (size);
|
||
memory->zalloc_p = 1;
|
||
memory->next = me->alloc_of_hw;
|
||
me->alloc_of_hw = memory;
|
||
return memory->alloc;
|
||
}
|
||
|
||
extern void *
|
||
hw_malloc (struct hw *me, unsigned long size)
|
||
{
|
||
struct hw_alloc_data *memory = ZALLOC (struct hw_alloc_data);
|
||
memory->alloc = zalloc (size);
|
||
memory->zalloc_p = 0;
|
||
memory->next = me->alloc_of_hw;
|
||
me->alloc_of_hw = memory;
|
||
return memory->alloc;
|
||
}
|
||
|
||
extern void
|
||
hw_free (struct hw *me,
|
||
void *alloc)
|
||
{
|
||
struct hw_alloc_data **memory;
|
||
for (memory = &me->alloc_of_hw;
|
||
*memory != NULL;
|
||
memory = &(*memory)->next)
|
||
{
|
||
if ((*memory)->alloc == alloc)
|
||
{
|
||
struct hw_alloc_data *die = (*memory);
|
||
(*memory) = die->next;
|
||
if (die->zalloc_p)
|
||
zfree (die->alloc);
|
||
else
|
||
free (die->alloc);
|
||
zfree (die);
|
||
return;
|
||
}
|
||
}
|
||
hw_abort (me, "free of memory not belonging to a device");
|
||
}
|
||
|
||
extern void
|
||
hw_free_all (struct hw *me)
|
||
{
|
||
while (me->alloc_of_hw != NULL)
|
||
{
|
||
hw_free (me, me->alloc_of_hw->alloc);
|
||
}
|
||
}
|