|  | /*  This file is part of the program psim. | 
|  |  | 
|  | Copyright (C) 1994-1996, 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 3 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, see <http://www.gnu.org/licenses/>. | 
|  |  | 
|  | */ | 
|  |  | 
|  |  | 
|  | #ifndef _DEVICE_TABLE_H_ | 
|  | #define _DEVICE_TABLE_H_ | 
|  |  | 
|  | #include "basics.h" | 
|  | #include "device.h" | 
|  | #include "tree.h" | 
|  |  | 
|  | #ifdef HAVE_STRING_H | 
|  | #include <string.h> | 
|  | #else | 
|  | #ifdef HAVE_STRINGS_H | 
|  | #include <strings.h> | 
|  | #endif | 
|  | #endif | 
|  |  | 
|  |  | 
|  | typedef struct _device_callbacks device_callbacks; | 
|  |  | 
|  |  | 
|  | /* The creator, returns a pointer to any data that should be allocated | 
|  | once during (multiple) simulation runs */ | 
|  |  | 
|  | typedef void *(device_creator) | 
|  | (const char *name, | 
|  | const device_unit *unit_address, | 
|  | const char *args); | 
|  |  | 
|  |  | 
|  | /* two stages of initialization */ | 
|  |  | 
|  | typedef void (device_init_callback) | 
|  | (device *me); | 
|  |  | 
|  | typedef struct _device_init_callbacks { | 
|  | device_init_callback *address; /* NULL - ignore */ | 
|  | device_init_callback *data; /* NULL - ignore */ | 
|  | } device_init_callbacks; | 
|  |  | 
|  |  | 
|  | /* attaching/detaching a devices address space to its parent */ | 
|  |  | 
|  | typedef void (device_address_callback) | 
|  | (device *me, | 
|  | attach_type attach, | 
|  | int space, | 
|  | unsigned_word addr, | 
|  | unsigned nr_bytes, | 
|  | access_type access, | 
|  | device *client); /*callback/default*/ | 
|  |  | 
|  | typedef struct _device_address_callbacks { | 
|  | device_address_callback *attach; | 
|  | device_address_callback *detach; | 
|  | } device_address_callbacks; | 
|  |  | 
|  |  | 
|  | /* I/O operations - from parent */ | 
|  |  | 
|  | typedef unsigned (device_io_read_buffer_callback) | 
|  | (device *me, | 
|  | void *dest, | 
|  | int space, | 
|  | unsigned_word addr, | 
|  | unsigned nr_bytes, | 
|  | cpu *processor, | 
|  | unsigned_word cia); | 
|  |  | 
|  | typedef unsigned (device_io_write_buffer_callback) | 
|  | (device *me, | 
|  | const void *source, | 
|  | int space, | 
|  | unsigned_word addr, | 
|  | unsigned nr_bytes, | 
|  | cpu *processor, | 
|  | unsigned_word cia); | 
|  |  | 
|  | typedef struct _device_io_callbacks { /* NULL - error */ | 
|  | device_io_read_buffer_callback *read_buffer; | 
|  | device_io_write_buffer_callback *write_buffer; | 
|  | } device_io_callbacks; | 
|  |  | 
|  |  | 
|  | /* DMA transfers by a device via its parent */ | 
|  |  | 
|  | typedef unsigned (device_dma_read_buffer_callback) | 
|  | (device *me, | 
|  | void *dest, | 
|  | int space, | 
|  | unsigned_word addr, | 
|  | unsigned nr_bytes); | 
|  |  | 
|  | typedef unsigned (device_dma_write_buffer_callback) | 
|  | (device *me, | 
|  | const void *source, | 
|  | int space, | 
|  | unsigned_word addr, | 
|  | unsigned nr_bytes, | 
|  | int violate_read_only_section); | 
|  |  | 
|  | typedef struct _device_dma_callbacks { /* NULL - error */ | 
|  | device_dma_read_buffer_callback *read_buffer; | 
|  | device_dma_write_buffer_callback *write_buffer; | 
|  | } device_dma_callbacks; | 
|  |  | 
|  |  | 
|  | /* Interrupts */ | 
|  |  | 
|  | typedef void (device_interrupt_event_callback) | 
|  | (device *me, | 
|  | int my_port, | 
|  | device *source, | 
|  | int source_port, | 
|  | int level, | 
|  | cpu *processor, | 
|  | unsigned_word cia); | 
|  |  | 
|  | typedef void (device_child_interrupt_event_callback) | 
|  | (device *me, | 
|  | device *parent, | 
|  | device *source, | 
|  | int source_port, | 
|  | int level, | 
|  | cpu *processor, | 
|  | unsigned_word cia); | 
|  |  | 
|  | typedef struct _device_interrupt_port_descriptor { | 
|  | const char *name; | 
|  | int number; | 
|  | int nr_ports; | 
|  | port_direction direction; | 
|  | } device_interrupt_port_descriptor; | 
|  |  | 
|  | typedef struct _device_interrupt_callbacks { | 
|  | device_interrupt_event_callback *event; | 
|  | device_child_interrupt_event_callback *child_event; | 
|  | const device_interrupt_port_descriptor *ports; | 
|  | } device_interrupt_callbacks; | 
|  |  | 
|  |  | 
|  | /* symbolic value decoding */ | 
|  |  | 
|  | typedef int (device_unit_decode_callback) | 
|  | (device *bus, | 
|  | const char *unit, | 
|  | device_unit *address); | 
|  |  | 
|  | typedef int (device_unit_encode_callback) | 
|  | (device *bus, | 
|  | const device_unit *unit_address, | 
|  | char *buf, | 
|  | int sizeof_buf); | 
|  |  | 
|  | typedef int (device_address_to_attach_address_callback) | 
|  | (device *bus, | 
|  | const device_unit *address, | 
|  | int *attach_space, | 
|  | unsigned_word *attach_address, | 
|  | device *client); | 
|  |  | 
|  | typedef int (device_size_to_attach_size_callback) | 
|  | (device *bus, | 
|  | const device_unit *size, | 
|  | unsigned *nr_bytes, | 
|  | device *client); | 
|  |  | 
|  | typedef struct _device_convert_callbacks { | 
|  | device_unit_decode_callback *decode_unit; | 
|  | device_unit_encode_callback *encode_unit; | 
|  | device_address_to_attach_address_callback *address_to_attach_address; | 
|  | device_size_to_attach_size_callback *size_to_attach_size; | 
|  | } device_convert_callbacks; | 
|  |  | 
|  |  | 
|  | /* instances */ | 
|  |  | 
|  | typedef void (device_instance_delete_callback) | 
|  | (device_instance *instance); | 
|  |  | 
|  | typedef int (device_instance_read_callback) | 
|  | (device_instance *instance, | 
|  | void *buf, | 
|  | unsigned_word len); | 
|  |  | 
|  | typedef int (device_instance_write_callback) | 
|  | (device_instance *instance, | 
|  | const void *buf, | 
|  | unsigned_word len); | 
|  |  | 
|  | typedef int (device_instance_seek_callback) | 
|  | (device_instance *instance, | 
|  | unsigned_word pos_hi, | 
|  | unsigned_word pos_lo); | 
|  |  | 
|  | typedef int (device_instance_method) | 
|  | (device_instance *instance, | 
|  | int n_stack_args, | 
|  | unsigned_cell stack_args[/*n_stack_args*/], | 
|  | int n_stack_returns, | 
|  | unsigned_cell stack_returns[/*n_stack_returns*/]); | 
|  |  | 
|  | typedef struct _device_instance_methods { | 
|  | const char *name; | 
|  | device_instance_method *method; | 
|  | } device_instance_methods; | 
|  |  | 
|  | struct _device_instance_callbacks { /* NULL - error */ | 
|  | device_instance_delete_callback *delete; | 
|  | device_instance_read_callback *read; | 
|  | device_instance_write_callback *write; | 
|  | device_instance_seek_callback *seek; | 
|  | const device_instance_methods *methods; | 
|  | }; | 
|  |  | 
|  | typedef device_instance *(device_create_instance_callback) | 
|  | (device *me, | 
|  | const char *full_path, | 
|  | const char *args); | 
|  |  | 
|  | typedef device_instance *(package_create_instance_callback) | 
|  | (device_instance *parent, | 
|  | const char *args); | 
|  |  | 
|  |  | 
|  | /* all else fails */ | 
|  |  | 
|  | typedef int (device_ioctl_callback) | 
|  | (device *me, | 
|  | cpu *processor, | 
|  | unsigned_word cia, | 
|  | device_ioctl_request request, | 
|  | va_list ap); | 
|  |  | 
|  | typedef void (device_usage_callback) | 
|  | (int verbose); | 
|  |  | 
|  |  | 
|  | /* the callbacks */ | 
|  |  | 
|  | struct _device_callbacks { | 
|  |  | 
|  | /* initialization */ | 
|  | device_init_callbacks init; | 
|  |  | 
|  | /* address/data config - from child */ | 
|  | device_address_callbacks address; | 
|  |  | 
|  | /* address/data transfer - from parent */ | 
|  | device_io_callbacks io; | 
|  |  | 
|  | /* address/data transfer - from child */ | 
|  | device_dma_callbacks dma; | 
|  |  | 
|  | /* interrupt signalling */ | 
|  | device_interrupt_callbacks interrupt; | 
|  |  | 
|  | /* bus address decoding */ | 
|  | device_convert_callbacks convert; | 
|  |  | 
|  | /* instances */ | 
|  | device_create_instance_callback *instance_create; | 
|  |  | 
|  | /* back door to anything we've forgot */ | 
|  | device_ioctl_callback *ioctl; | 
|  | device_usage_callback *usage; | 
|  | }; | 
|  |  | 
|  |  | 
|  | /* Table of all the devices and a function to lookup/create a device | 
|  | from its name */ | 
|  |  | 
|  | typedef struct _device_descriptor device_descriptor; | 
|  | struct _device_descriptor { | 
|  | const char *name; | 
|  | device_creator *creator; | 
|  | const device_callbacks *callbacks; | 
|  | }; | 
|  |  | 
|  | extern const device_descriptor *const device_table[]; | 
|  | #include "hw.h" | 
|  |  | 
|  |  | 
|  | /* Pass through, ignore and generic callback functions.  A call going | 
|  | towards the root device are passed on up, local calls are ignored | 
|  | and call downs abort */ | 
|  |  | 
|  | extern device_address_callback passthrough_device_address_attach; | 
|  | extern device_address_callback passthrough_device_address_detach; | 
|  | extern device_dma_read_buffer_callback passthrough_device_dma_read_buffer; | 
|  | extern device_dma_write_buffer_callback passthrough_device_dma_write_buffer; | 
|  |  | 
|  | extern device_unit_decode_callback ignore_device_unit_decode; | 
|  |  | 
|  | extern device_init_callback generic_device_init_address; | 
|  | extern device_unit_decode_callback generic_device_unit_decode; | 
|  | extern device_unit_encode_callback generic_device_unit_encode; | 
|  | extern device_address_to_attach_address_callback generic_device_address_to_attach_address; | 
|  | extern device_size_to_attach_size_callback generic_device_size_to_attach_size; | 
|  |  | 
|  |  | 
|  | extern const device_callbacks passthrough_device_callbacks; | 
|  |  | 
|  | #endif /* _DEVICE_TABLE_H_ */ |