mame/src/emu/output.c
Aaron Giles ad4910a8a8 Bulk change alert.
This update changes the way we handle memory allocation. Rather
than allocating in terms of bytes, allocations are now done in
terms of objects. This is done via new set of macros that replace
the malloc_or_die() macro:

  alloc_or_die(t) - allocate memory for an object of type 't'
  alloc_array_or_die(t,c) - allocate memory for an array of 'c' objects of type 't'
  alloc_clear_or_die(t) - same as alloc_or_die but memset's the memory to 0
  alloc_array_clear_or_die(t,c) - same as alloc_array_or_die but memset's the memory to 0

All original callers of malloc_or_die have been updated to call these
new macros. If you just need an array of bytes, you can use
alloc_array_or_die(UINT8, numbytes).

Made a similar change to the auto_* allocation macros. In addition,
added 'machine' as a required parameter to the auto-allocation macros,
as the resource pools will eventually be owned by the machine object.
The new macros are:

  auto_alloc(m,t) - allocate memory for an object of type 't'
  auto_alloc_array(m,t,c) - allocate memory for an array of 'c' objects of type 't'
  auto_alloc_clear(m,t) - allocate and memset
  auto_alloc_array_clear(m,t,c) - allocate and memset

All original calls or auto_malloc have been updated to use the new
macros. In addition, auto_realloc(), auto_strdup(), auto_astring_alloc(),
and auto_bitmap_alloc() have been updated to take a machine parameter.

Changed validity check allocations to not rely on auto_alloc* anymore
because they are not done in the context of a machine.

One final change that is included is the removal of SMH_BANKn macros.
Just use SMH_BANK(n) instead, which is what the previous macros mapped
to anyhow.
2009-04-26 23:54:37 +00:00

407 lines
10 KiB
C

/***************************************************************************
output.c
General purpose output routines.
Copyright Nicola Salmoria and the MAME Team.
Visit http://mamedev.org for licensing and usage restrictions.
***************************************************************************/
#include "output.h"
#include "mame.h"
#include <zlib.h>
/***************************************************************************
CONSTANTS
***************************************************************************/
#define HASH_SIZE 53
/***************************************************************************
TYPE DEFINITIONS
***************************************************************************/
typedef struct _output_notify output_notify;
struct _output_notify
{
output_notify * next; /* link to next item */
output_notifier_func notifier; /* callback to call */
void * param; /* parameter to pass the callback */
};
typedef struct _output_item output_item;
struct _output_item
{
output_item * next; /* next item in list */
const char * name; /* string name of the item */
UINT32 hash; /* hash for this item name */
UINT32 id; /* unique ID for this item */
INT32 value; /* current value */
output_notify * notifylist; /* list of notifier callbacks */
};
/***************************************************************************
GLOBAL VARIABLES
***************************************************************************/
static output_item *itemtable[HASH_SIZE];
static output_notify *global_notifylist;
static UINT32 uniqueid = 12345;
/***************************************************************************
FUNCTION PROTOTYPES
***************************************************************************/
static void output_pause(running_machine *machine, int pause);
static void output_exit(running_machine *machine);
/***************************************************************************
INLINE FUNCTIONS
***************************************************************************/
/*-------------------------------------------------
copy_string - make a copy of a string
-------------------------------------------------*/
INLINE const char *copy_string(const char *string)
{
char *newstring = alloc_array_or_die(char, strlen(string) + 1);
strcpy(newstring, string);
return newstring;
}
/*-------------------------------------------------
get_hash - return the hash of an output value
-------------------------------------------------*/
INLINE UINT32 get_hash(const char *string)
{
return crc32(0, (UINT8 *)string, (UINT32)strlen(string));
}
/*-------------------------------------------------
find_item - find an item based on a string
-------------------------------------------------*/
INLINE output_item *find_item(const char *string)
{
UINT32 hash = get_hash(string);
output_item *item;
/* use the hash as a starting point and find an entry */
for (item = itemtable[hash % HASH_SIZE]; item != NULL; item = item->next)
if (item->hash == hash && strcmp(string, item->name) == 0)
return item;
return NULL;
}
/*-------------------------------------------------
create_new_item - create a new item
-------------------------------------------------*/
INLINE output_item *create_new_item(const char *outname, INT32 value)
{
output_item *item = alloc_or_die(output_item);
UINT32 hash = get_hash(outname);
/* fill in the data */
item->next = itemtable[hash % HASH_SIZE];
item->name = copy_string(outname);
item->hash = hash;
item->id = uniqueid++;
item->value = value;
item->notifylist = NULL;
/* add us to the hash table */
itemtable[hash % HASH_SIZE] = item;
return item;
}
/***************************************************************************
CORE IMPLEMENTATION
***************************************************************************/
/*-------------------------------------------------
output_init - initialize everything
-------------------------------------------------*/
void output_init(running_machine *machine)
{
/* add pause callback */
add_pause_callback(machine, output_pause);
/* get a callback when done */
add_exit_callback(machine, output_exit);
/* reset the lists */
memset(itemtable, 0, sizeof(itemtable));
global_notifylist = NULL;
}
/*-------------------------------------------------
output_pause - send pause message
-------------------------------------------------*/
static void output_pause(running_machine *machine, int pause)
{
output_set_value("pause", pause & 1);
}
/*-------------------------------------------------
output_exit - cleanup on exit
-------------------------------------------------*/
static void output_exit(running_machine *machine)
{
output_notify *notify;
output_item *item;
int hash;
/* remove all items */
for (hash = 0; hash < HASH_SIZE; hash++)
for (item = itemtable[hash]; item != NULL; )
{
output_item *next = item->next;
/* remove all notifiers */
for (notify = item->notifylist; notify != NULL; )
{
output_notify *next = notify->next;
free(notify);
notify = next;
}
/* free the name and the item */
if (item->name != NULL)
free((void *)item->name);
free(item);
item = next;
}
/* remove all global notifiers */
for (notify = global_notifylist; notify != NULL; )
{
output_notify *next = notify->next;
free(notify);
notify = next;
}
}
/*-------------------------------------------------
output_set_value - set the value of an output
-------------------------------------------------*/
void output_set_value(const char *outname, INT32 value)
{
output_item *item = find_item(outname);
output_notify *notify;
INT32 oldval;
/* if no item of that name, create a new one and send the item's state */
if (item == NULL)
{
item = create_new_item(outname, value);
oldval = value + 1;
}
else
{
/* set the new value */
oldval = item->value;
item->value = value;
}
/* if the value is different, signal the notifier */
if (oldval != value)
{
/* call the local notifiers first */
for (notify = item->notifylist; notify != NULL; notify = notify->next)
(*notify->notifier)(outname, value, notify->param);
/* call the global notifiers next */
for (notify = global_notifylist; notify != NULL; notify = notify->next)
(*notify->notifier)(outname, value, notify->param);
}
}
/*-------------------------------------------------
output_set_indexed_value - set the value of an
indexed output
-------------------------------------------------*/
void output_set_indexed_value(const char *basename, int index, int value)
{
char buffer[100];
char *dest = buffer;
/* copy the string */
while (*basename != 0)
*dest++ = *basename++;
/* append the index */
if (index >= 1000) *dest++ = '0' + ((index / 1000) % 10);
if (index >= 100) *dest++ = '0' + ((index / 100) % 10);
if (index >= 10) *dest++ = '0' + ((index / 10) % 10);
*dest++ = '0' + (index % 10);
*dest++ = 0;
/* set the value */
output_set_value(buffer, value);
}
/*-------------------------------------------------
output_get_value - return the value of an
output
-------------------------------------------------*/
INT32 output_get_value(const char *outname)
{
output_item *item = find_item(outname);
/* if no item, value is 0 */
if (item == NULL)
return 0;
return item->value;
}
/*-------------------------------------------------
output_get_indexed_value - get the value of an
indexed output
-------------------------------------------------*/
INT32 output_get_indexed_value(const char *basename, int index)
{
char buffer[100];
char *dest = buffer;
/* copy the string */
while (*basename != 0)
*dest++ = *basename++;
/* append the index */
if (index >= 1000) *dest++ = '0' + ((index / 1000) % 10);
if (index >= 100) *dest++ = '0' + ((index / 100) % 10);
if (index >= 10) *dest++ = '0' + ((index / 10) % 10);
*dest++ = '0' + (index % 10);
*dest++ = 0;
/* set the value */
return output_get_value(buffer);
}
/*-------------------------------------------------
output_set_notifier - sets a notifier callback
for a particular output, or for all outputs
if NULL is specified
-------------------------------------------------*/
void output_set_notifier(const char *outname, output_notifier_func callback, void *param)
{
output_notify **headptr;
/* if an item is specified, find it */
if (outname != NULL)
{
output_item *item = find_item(outname);
/* if no item of that name, create a new one */
if (item == NULL)
item = create_new_item(outname, 0);
headptr = &item->notifylist;
}
/* if no item is specified, we add to the global list */
else
headptr = &global_notifylist;
/* find the end of the list and add to it */
while (*headptr != NULL)
headptr = &(*headptr)->next;
*headptr = alloc_or_die(output_notify);
/* fill in the new record */
(*headptr)->next = NULL;
(*headptr)->notifier = callback;
(*headptr)->param = param;
}
/*-------------------------------------------------
output_notify_all - immediately call the given
notifier for all outputs
-------------------------------------------------*/
void output_notify_all(output_notifier_func callback, void *param)
{
output_item *item;
int hash;
/* remove all items */
for (hash = 0; hash < HASH_SIZE; hash++)
for (item = itemtable[hash]; item != NULL; item = item->next)
(*callback)(item->name, item->value, param);
}
/*-------------------------------------------------
output_name_to_id - returns a unique ID for
a given name
-------------------------------------------------*/
UINT32 output_name_to_id(const char *outname)
{
output_item *item = find_item(outname);
/* if no item, ID is 0 */
if (item == NULL)
return 0;
return item->id;
}
/*-------------------------------------------------
output_id_to_name - returns a name that maps
to a given unique ID
-------------------------------------------------*/
const char *output_id_to_name(UINT32 id)
{
output_item *item;
int hash;
/* remove all items */
for (hash = 0; hash < HASH_SIZE; hash++)
for (item = itemtable[hash]; item != NULL; item = item->next)
if (item->id == id)
return item->name;
/* nothing found, return NULL */
return NULL;
}