mame/src/emu/softlist.c

1257 lines
40 KiB
C

/***************************************************************************
softlist.c
Software list construction helpers.
***************************************************************************/
#include "emu.h"
#include "pool.h"
#include "emuopts.h"
#include "softlist.h"
#include "clifront.h"
#include "validity.h"
#include "expat.h"
#include <ctype.h>
//**************************************************************************
// TYPE DEFINITIONS
//**************************************************************************
typedef tagmap_t<software_info *> softlist_map;
// ======================> softlist_parser
class softlist_parser
{
public:
// construction (== execution)
softlist_parser(software_list_device &list, astring &errors);
private:
enum parse_position
{
POS_ROOT,
POS_MAIN,
POS_SOFT,
POS_PART,
POS_DATA
};
// internal parsing helpers
const char *filename() const { return m_list.filename(); }
const char *infoname() const { return (m_current_info != NULL) ? m_current_info->shortname() : "???"; }
int line() const { return XML_GetCurrentLineNumber(m_parser); }
int column() const { return XML_GetCurrentColumnNumber(m_parser); }
const char *parser_error() const { return XML_ErrorString(XML_GetErrorCode(m_parser)); }
// internal error helpers
void ATTR_PRINTF(2,3) parse_error(const char *fmt, ...);
void unknown_tag(const char *tagname) { parse_error("Unknown tag: %s", tagname); }
void unknown_attribute(const char *attrname) { parse_error("Unknown attribute: %s", attrname); }
// internal helpers
void parse_attributes(const char **attributes, int numattrs, const char *attrlist[], const char *outlist[]);
void add_rom_entry(const char *name, const char *hashdata, UINT32 offset, UINT32 length, UINT32 flags);
// expat callbacks
static void *expat_malloc(size_t size);
static void *expat_realloc(void *ptr, size_t size);
static void expat_free(void *ptr);
static void start_handler(void *data, const char *tagname, const char **attributes);
static void data_handler(void *data, const XML_Char *s, int len);
static void end_handler(void *data, const char *name);
// internal parsing
void parse_root_start(const char *tagname, const char **attributes);
void parse_main_start(const char *tagname, const char **attributes);
void parse_soft_start(const char *tagname, const char **attributes);
void parse_part_start(const char *tagname, const char **attributes);
void parse_data_start(const char *tagname, const char **attributes);
void parse_soft_end(const char *name);
// internal parsing state
software_list_device & m_list;
astring & m_errors;
XML_Parser m_parser;
bool m_done;
bool m_data_accum_expected;
astring m_data_accum;
software_info * m_current_info;
software_part * m_current_part;
parse_position m_pos;
};
//**************************************************************************
// GLOBAL VARIABLES
//**************************************************************************
// device type definition
const device_type SOFTWARE_LIST = &device_creator<software_list_device>;
//**************************************************************************
// SOFTWARE PART
//**************************************************************************
//-------------------------------------------------
// software_part - constructor
//-------------------------------------------------
software_part::software_part(software_info &info, const char *name, const char *interface)
: m_next(NULL),
m_info(info),
m_name(name),
m_interface(interface)
{
// ensure strings we are passed are in the string pool
assert(info.list().string_pool_contains(name));
assert(info.list().string_pool_contains(interface));
}
//-------------------------------------------------
// feature - return the value of the given
// feature, if specified
//-------------------------------------------------
const char *software_part::feature(const char *feature_name) const
{
assert(feature_name != NULL);
// scan the feature list for an entry matching feature_name and return the value
for (const feature_list_item *feature = m_featurelist.first(); feature != NULL; feature = feature->next())
if (strcmp(feature->name(), feature_name) == 0)
return feature->value();
return NULL;
}
//-------------------------------------------------
// is_compatible - determine if we are compatible
// with the given software_list_device
//-------------------------------------------------
bool software_part::is_compatible(const software_list_device &swlistdev) const
{
// get the compatibility feature and the softlist filter; if either is NULL, assume compatible
const char *compatibility = feature("compatibility");
const char *filter = swlistdev.filter();
if (compatibility == NULL || filter == NULL)
return true;
// copy the comma-delimited strings and ensure they end with a final comma
astring comp = astring(compatibility).cat(",");
astring filt = astring(filter).cat(",");
// iterate over filter items and see if they exist in the compatibility list; if so, return true
for (int start = 0, end = filt.chr(start, ','); end != -1; start = end + 1, end = filt.chr(start, ','))
{
astring token(filt, start, end - start + 1);
if (comp.find(0, token.c_str()) != -1)
return true;
}
return false;
}
//-------------------------------------------------
// matches_interface - determine if we match
// an interface in the provided list
//-------------------------------------------------
bool software_part::matches_interface(const char *interface_list) const
{
// if we have no interface, then we match by default
if (m_interface == NULL)
return true;
// copy the comma-delimited interface list and ensure it ends with a final comma
astring interfaces = astring(interface_list).cat(",");
// then add a comma to the end of our interface and return true if we find it in the list string
astring our_interface = astring(m_interface).cat(",");
return (interfaces.find(0, our_interface.c_str()) != -1);
}
//**************************************************************************
// SOFTWARE INFO
//**************************************************************************
//-------------------------------------------------
// software_info - constructor
//-------------------------------------------------
software_info::software_info(software_list_device &list, const char *name, const char *parent, const char *supported)
: m_next(NULL),
m_list(list),
m_supported(SOFTWARE_SUPPORTED_YES),
m_shortname(name),
m_longname(NULL),
m_parentname(parent),
m_year(NULL),
m_publisher(NULL)
{
// ensure strings we are passed are in the string pool
assert(list.string_pool_contains(name));
assert(list.string_pool_contains(parent));
// handle the supported flag if provided
if (supported != NULL)
{
if (strcmp(supported, "partial") == 0)
m_supported = SOFTWARE_SUPPORTED_PARTIAL;
else if (strcmp(supported, "no") == 0)
m_supported = SOFTWARE_SUPPORTED_NO;
}
}
//-------------------------------------------------
// find_part - find a part by name with an
// optional interface match
//-------------------------------------------------
software_part *software_info::find_part(const char *partname, const char *interface)
{
// if neither partname nor interface supplied, then we just return the first entry
if (partname != NULL && strlen(partname)==0) partname = NULL;
if (partname == NULL && interface == NULL)
return m_partdata.first();
// look for the part by name and match against the interface if provided
for (software_part *part = m_partdata.first(); part != NULL; part = part->next())
if (partname != NULL && strcmp(partname, part->name()) == 0)
{
if (interface == NULL || part->matches_interface(interface))
return part;
}
else if (partname == NULL && part->matches_interface(interface))
return part;
return NULL;
}
//-------------------------------------------------
// has_multiple_parts - return true if we have
// more than one part matching the given
// interface
//-------------------------------------------------
bool software_info::has_multiple_parts(const char *interface) const
{
int count = 0;
// increment the count for each match and stop if we hit more than 1
for (software_part *part = first_part(); part != NULL; part = part->next())
if (part->matches_interface(interface))
if (++count > 1)
return true;
return false;
}
//**************************************************************************
// SOFTWARE LIST DEVICE
//**************************************************************************
//-------------------------------------------------
// software_list_device - constructor
//-------------------------------------------------
software_list_device::software_list_device(const machine_config &mconfig, const char *tag, device_t *owner, UINT32 clock)
: device_t(mconfig, SOFTWARE_LIST, "Software list", tag, owner, clock, "software_list", __FILE__),
m_list_type(SOFTWARE_LIST_ORIGINAL_SYSTEM),
m_filter(NULL),
m_parsed(false),
m_file(mconfig.options().hash_path(), OPEN_FLAG_READ)
{
}
//-------------------------------------------------
// static_set_type - configuration helper
// to set the list type
//-------------------------------------------------
void software_list_device::static_set_type(device_t &device, const char *list, softlist_type list_type)
{
software_list_device &swlistdev = downcast<software_list_device &>(device);
swlistdev.m_list_name.cpy(list);
swlistdev.m_list_type = list_type;
}
//-------------------------------------------------
// static_set_custom_handler - configuration
// helper to set a custom callback
//-------------------------------------------------
void software_list_device::static_set_filter(device_t &device, const char *filter)
{
downcast<software_list_device &>(device).m_filter = filter;
}
//-------------------------------------------------
// device_start - device-specific startup
//-------------------------------------------------
void software_list_device::device_start()
{
}
//-------------------------------------------------
// find_approx_matches - search ourselves for
// a list of possible matches of the given name
// and optional interface
//-------------------------------------------------
void software_list_device::find_approx_matches(const char *name, int matches, software_info **list, const char *interface)
{
// if no name, return
if (name == NULL || name[0] == 0)
return;
// initialize everyone's states
dynamic_array<int> penalty(matches);
for (int matchnum = 0; matchnum < matches; matchnum++)
{
penalty[matchnum] = 9999;
list[matchnum] = NULL;
}
// iterate over our info (will cause a parse if needed)
for (software_info *swinfo = first_software_info(); swinfo != NULL; swinfo = swinfo->next())
{
software_part *part = swinfo->first_part();
if ((interface == NULL || part->matches_interface(interface)) && part->is_compatible(*this))
{
// pick the best match between driver name and description
int longpenalty = driver_list::penalty_compare(name, swinfo->longname());
int shortpenalty = driver_list::penalty_compare(name, swinfo->shortname());
int curpenalty = MIN(longpenalty, shortpenalty);
// insert into the sorted table of matches
for (int matchnum = matches - 1; matchnum >= 0; matchnum--)
{
// stop if we're worse than the current entry
if (curpenalty >= penalty[matchnum])
break;
// as long as this isn't the last entry, bump this one down
if (matchnum < matches - 1)
{
penalty[matchnum + 1] = penalty[matchnum];
list[matchnum + 1] = list[matchnum];
}
list[matchnum] = swinfo;
penalty[matchnum] = curpenalty;
}
}
}
}
//-------------------------------------------------
// release - reset to a pre-parsed state
//-------------------------------------------------
void software_list_device::release()
{
osd_printf_verbose("Resetting %s\n", m_file.filename());
m_parsed = false;
m_description = NULL;
m_errors.reset();
m_infolist.reset();
m_stringpool.reset();
}
//-------------------------------------------------
// find_by_name - find a software list by name
// across all software list devices
//-------------------------------------------------
software_list_device *software_list_device::find_by_name(const machine_config &config, const char *name)
{
// iterate over each device in the system and find a match
software_list_device_iterator deviter(config.root_device());
for (software_list_device *swlistdev = deviter.first(); swlistdev != NULL; swlistdev = deviter.next())
if (strcmp(swlistdev->list_name(), name) == 0)
return swlistdev;
return NULL;
}
//-------------------------------------------------
// software_display_matches - display a list of
// possible matches in the system to the given
// name, across all software list devices
//-------------------------------------------------
void software_list_device::display_matches(const machine_config &config, const char *interface, const char *name)
{
// check if there is at least one software list
software_list_device_iterator deviter(config.root_device());
if (deviter.first())
osd_printf_error("\n\"%s\" approximately matches the following\n"
"supported software items (best match first):\n\n", name);
// iterate through lists
for (software_list_device *swlistdev = deviter.first(); swlistdev != NULL; swlistdev = deviter.next())
{
// get the top 16 approximate matches for the selected device interface (i.e. only carts for cartslot, etc.)
software_info *matches[16] = { 0 };
swlistdev->find_approx_matches(name, ARRAY_LENGTH(matches), matches, interface);
// if we found some, print them
if (matches[0] != 0)
{
// different output depending on original system or compatible
if (swlistdev->list_type() == SOFTWARE_LIST_ORIGINAL_SYSTEM)
osd_printf_error("* Software list \"%s\" (%s) matches: \n", swlistdev->list_name(), swlistdev->description());
else
osd_printf_error("* Compatible software list \"%s\" (%s) matches: \n", swlistdev->list_name(), swlistdev->description());
// print them out
for (int softnum = 0; softnum < ARRAY_LENGTH(matches); softnum++)
if (matches[softnum] != NULL)
osd_printf_error("%-18s%s\n", matches[softnum]->shortname(), matches[softnum]->longname());
osd_printf_error("\n");
}
}
}
//-------------------------------------------------
// find - find an item by name in the software
// list, using wildcards and optionally starting
// from an intermediate point
//-------------------------------------------------
software_info *software_list_device::find(const char *look_for, software_info *prev)
{
// NULL search returns nothing
if (look_for == NULL)
return NULL;
bool iswild = strchr(look_for, '*') != NULL || strchr(look_for, '?');
// find a match (will cause a parse if needed when calling first_software_info)
for (prev = (prev != NULL) ? prev->next() : first_software_info(); prev != NULL; prev = prev->next())
if ((iswild && core_strwildcmp(look_for, prev->shortname()) == 0) || core_stricmp(look_for, prev->shortname()) == 0)
break;
return prev;
}
//-------------------------------------------------
// parse - parse our softlist file
//-------------------------------------------------
void software_list_device::parse()
{
// skip if done
if (m_parsed)
return;
// reset the errors
m_errors.reset();
// attempt to open the file
file_error filerr = m_file.open(m_list_name.c_str(), ".xml");
if (filerr == FILERR_NONE)
{
// parse if no error
softlist_parser parser(*this, m_errors);
m_file.close();
}
else
m_errors.printf("Error opening file: %s\n", filename());
// indicate that we've been parsed
m_parsed = true;
}
//-------------------------------------------------
// device_validity_check - validate the device
// configuration
//-------------------------------------------------
void software_list_device::device_validity_check(validity_checker &valid) const
{
// add to the global map whenever we check a list so we don't re-check
// it in the future
if (valid.already_checked(astring("softlist/").cat(m_list_name.c_str()).c_str()))
return;
// do device validation only in case of validate command
if (strcmp(mconfig().options().command(), CLICOMMAND_VALIDATE) != 0)
return;
// actually do the validate
const_cast<software_list_device *>(this)->internal_validity_check(valid);
}
//-------------------------------------------------
// internal_validity_check - internal helper to
// check the list
//-------------------------------------------------
void software_list_device::internal_validity_check(validity_checker &valid)
{
enum { NAME_LEN_PARENT = 8, NAME_LEN_CLONE = 16 };
softlist_map names;
softlist_map descriptions;
for (software_info *swinfo = first_software_info(); swinfo != NULL; swinfo = swinfo->next())
{
// first parse and output core errors if any
if (m_errors.len() > 0)
{
osd_printf_error("%s: Errors parsing software list:\n%s", filename(), errors_string());
break;
}
// Now check if the xml data is valid:
// Did we lost any description?
if (swinfo->longname() == NULL)
{
osd_printf_error("%s: %s has no description\n", filename(), swinfo->shortname());
break;
}
// Did we lost any year?
if (swinfo->year() == NULL)
{
osd_printf_error("%s: %s has no year\n", filename(), swinfo->shortname());
break;
}
// Did we lost any publisher?
if (swinfo->publisher() == NULL)
{
osd_printf_error("%s: %s has no publisher\n", filename(), swinfo->shortname());
break;
}
// Second, since the xml is fine, run additional checks:
// check for duplicate names
if (names.add(swinfo->shortname(), swinfo, false) == TMERR_DUPLICATE)
{
software_info *match = names.find(swinfo->shortname());
osd_printf_error("%s: %s is a duplicate name (%s)\n", filename(), swinfo->shortname(), match->shortname());
}
// check for duplicate descriptions
if (descriptions.add(astring(swinfo->longname()).makelower().c_str(), swinfo, false) == TMERR_DUPLICATE)
osd_printf_error("%s: %s is a duplicate description (%s)\n", filename(), swinfo->longname(), swinfo->shortname());
bool is_clone = false;
if (swinfo->parentname() != NULL)
{
is_clone = true;
if (strcmp(swinfo->parentname(), swinfo->shortname()) == 0)
{
osd_printf_error("%s: %s is set as a clone of itself\n", filename(), swinfo->shortname());
break;
}
// make sure the parent exists
software_info *swinfo2 = find(swinfo->parentname());
if (swinfo2 == NULL)
osd_printf_error("%s: parent '%s' software for '%s' not found\n", filename(), swinfo->parentname(), swinfo->shortname());
else if (swinfo2->parentname() != NULL)
osd_printf_error("%s: %s is a clone of a clone\n", filename(), swinfo->shortname());
}
// make sure the driver name is 8 chars or less
if ((is_clone && strlen(swinfo->shortname()) > NAME_LEN_CLONE) || (!is_clone && strlen(swinfo->shortname()) > NAME_LEN_PARENT))
osd_printf_error("%s: %s %s driver name must be %d characters or less\n", filename(), swinfo->shortname(),
is_clone ? "clone" : "parent", is_clone ? NAME_LEN_CLONE : NAME_LEN_PARENT);
// make sure the year is only digits, '?' or '+'
for (const char *s = swinfo->year(); *s != 0; s++)
if (!isdigit((UINT8)*s) && *s != '?' && *s != '+')
{
osd_printf_error("%s: %s has an invalid year '%s'\n", filename(), swinfo->shortname(), swinfo->year());
break;
}
softlist_map part_names;
for (software_part *part = swinfo->first_part(); part != NULL; part = part->next())
{
if (part->interface() == NULL)
osd_printf_error("%s: %s has a part (%s) without interface\n", filename(), swinfo->shortname(), part->name());
if (part->romdata() == NULL)
osd_printf_error("%s: %s has a part (%s) with no data\n", filename(), swinfo->shortname(), part->name());
if (part_names.add(part->name(), swinfo, false) == TMERR_DUPLICATE)
osd_printf_error("%s: %s has a part (%s) whose name is duplicate\n", filename(), swinfo->shortname(), part->name());
for (const rom_entry *data = part->romdata(); data->_name != NULL; data++)
if (data->_hashdata != NULL)
{
// make sure it's all lowercase
for (const char *str = data->_name; *str; str++)
if (tolower((UINT8)*str) != *str)
{
osd_printf_error("%s: %s has upper case ROM name %s\n", filename(), swinfo->shortname(), data->_name);
break;
}
// make sure the hash is valid
hash_collection hashes;
if (!hashes.from_internal_string(data->_hashdata))
osd_printf_error("%s: %s has rom '%s' with an invalid hash string '%s'\n", filename(), swinfo->shortname(), data->_name, data->_hashdata);
}
}
}
// release all the memory
release();
}
//**************************************************************************
// SOFTWARE LIST PARSER
//**************************************************************************
//-------------------------------------------------
// softlist_parser - constructor
//-------------------------------------------------
softlist_parser::softlist_parser(software_list_device &list, astring &errors)
: m_list(list),
m_errors(errors),
m_done(false),
m_data_accum_expected(false),
m_current_info(NULL),
m_current_part(NULL),
m_pos(POS_ROOT)
{
osd_printf_verbose("Parsing %s\n", m_list.m_file.filename());
// set up memory callbacks
XML_Memory_Handling_Suite memcallbacks;
memcallbacks.malloc_fcn = expat_malloc;
memcallbacks.realloc_fcn = expat_realloc;
memcallbacks.free_fcn = expat_free;
// create the parser
m_parser = XML_ParserCreate_MM(NULL, &memcallbacks, NULL);
if (m_parser == NULL)
throw std::bad_alloc();
// set the handlers
XML_SetUserData(m_parser, this);
XML_SetElementHandler(m_parser, &softlist_parser::start_handler, &softlist_parser::end_handler);
XML_SetCharacterDataHandler(m_parser, &softlist_parser::data_handler);
// parse the file contents
m_list.m_file.seek(0, SEEK_SET);
char buffer[1024];
while (!m_done)
{
UINT32 length = m_list.m_file.read(buffer, sizeof(buffer));
m_done = m_list.m_file.eof();
if (XML_Parse(m_parser, buffer, length, m_done) == XML_STATUS_ERROR)
{
parse_error("%s", parser_error());
break;
}
}
// free the parser
XML_ParserFree(m_parser);
osd_printf_verbose("Parsing complete\n");
}
//-------------------------------------------------
// expat_malloc/expat_realloc/expat_free -
// wrappers for memory allocation functions so
// that they pass through out memory tracking
// systems
//-------------------------------------------------
void *softlist_parser::expat_malloc(size_t size)
{
return global_alloc_array_clear(UINT8, size);
}
void *softlist_parser::expat_realloc(void *ptr, size_t size)
{
if (ptr != NULL) global_free_array((UINT8 *)ptr);
return global_alloc_array_clear(UINT8, size);
}
void softlist_parser::expat_free(void *ptr)
{
global_free_array((UINT8 *)ptr);
}
//-------------------------------------------------
// parse_error - append a parsing error with
// filename, line and column information
//-------------------------------------------------
void ATTR_PRINTF(2,3) softlist_parser::parse_error(const char *fmt, ...)
{
// always start with filename(line.column):
m_errors.catprintf("%s(%d.%d): ", filename(), line(), column());
// append the remainder of the string
va_list va;
va_start(va, fmt);
m_errors.catvprintf(fmt, va);
va_end(va);
// append a newline at the end
m_errors.cat("\n");
}
//-------------------------------------------------
// parse_attributes - helper to parse a set of
// attributes into a list of strings
//-------------------------------------------------
void softlist_parser::parse_attributes(const char **attributes, int numattrs, const char *attrlist[], const char *outlist[])
{
// iterate over attribute/value pairs
for( ; attributes[0]; attributes += 2)
{
int index;
// look for a match among the attributes provided
for (index = 0; index < numattrs; index++)
if (strcmp(attributes[0], attrlist[index]) == 0)
{
// if found, set the corresponding output entry to the value
outlist[index] = attributes[1];
break;
}
// if not found, report an unknown attribute
if (index == numattrs)
unknown_attribute(attributes[0]);
}
}
//-------------------------------------------------
// add_rom_entry - append a new ROM entry to the
// current part's list
//-------------------------------------------------
void softlist_parser::add_rom_entry(const char *name, const char *hashdata, UINT32 offset, UINT32 length, UINT32 flags)
{
// get the current part
if (m_current_part == NULL)
{
parse_error("ROM entry added in invalid context");
return;
}
// make sure we don't add duplicate regions
if (name != NULL && (flags & ROMENTRY_TYPEMASK) == ROMENTRYTYPE_REGION)
for (int romentry = 0; romentry < m_current_part->m_romdata.count(); romentry++)
if (m_current_part->m_romdata[romentry]._name != NULL && strcmp(m_current_part->m_romdata[romentry]._name, name) == 0)
parse_error("Duplicated dataarea %s in software %s", name, infoname());
// create the new entry and append it
rom_entry &entry = m_current_part->m_romdata.append();
entry._name = m_list.add_string(name);
entry._hashdata = m_list.add_string(hashdata);
entry._offset = offset;
entry._length = length;
entry._flags = flags;
}
//-------------------------------------------------
// start_handler - expat handler for tag start
//-------------------------------------------------
void softlist_parser::start_handler(void *data, const char *tagname, const char **attributes)
{
// switch off the current state
softlist_parser *state = reinterpret_cast<softlist_parser *>(data);
switch (state->m_pos)
{
case POS_ROOT:
state->parse_root_start(tagname, attributes);
break;
case POS_MAIN:
state->parse_main_start(tagname, attributes);
break;
case POS_SOFT:
state->parse_soft_start(tagname, attributes);
break;
case POS_PART:
state->parse_part_start(tagname, attributes);
break;
case POS_DATA:
state->parse_data_start(tagname, attributes);
break;
}
// increment the state since this is a tag start
state->m_pos = parse_position(state->m_pos + 1);
}
//-------------------------------------------------
// end_handler - handle end-of-tag post-processing
//-------------------------------------------------
void softlist_parser::end_handler(void *data, const char *name)
{
// reset the text destination and bump the position down
softlist_parser *state = reinterpret_cast<softlist_parser *>(data);
state->m_pos = parse_position(state->m_pos - 1);
// switch off of the new position
switch (state->m_pos)
{
case POS_ROOT:
break;
case POS_MAIN:
state->m_current_info = NULL;
break;
case POS_SOFT:
state->parse_soft_end(name);
state->m_current_part = NULL;
break;
case POS_PART:
break;
case POS_DATA:
break;
}
// stop accumulating
state->m_data_accum_expected = false;
state->m_data_accum.reset();
}
//-------------------------------------------------
// data_handler - expat data handler
//-------------------------------------------------
void softlist_parser::data_handler(void *data, const XML_Char *s, int len)
{
softlist_parser *state = reinterpret_cast<softlist_parser *>(data);
// if we have an astring to accumulate data in, do it
if (state->m_data_accum_expected)
state->m_data_accum.cat(s, len);
// otherwise, report an error if the data is non-blank
else
for (int i = 0; i < len; i++)
if (!isspace(s[i]))
{
state->parse_error("Unexpected content");
break;
}
}
//-------------------------------------------------
// parse_root_start - handle tag start at the root
//-------------------------------------------------
void softlist_parser::parse_root_start(const char *tagname, const char **attributes)
{
// <softwarelist name='' description=''>
if (strcmp(tagname, "softwarelist") == 0)
{
static const char *attrnames[] = { "name", "description" };
const char *attrvalues[ARRAY_LENGTH(attrnames)] = { 0 };
parse_attributes(attributes, ARRAY_LENGTH(attrnames), attrnames, attrvalues);
if (attrvalues[1] != NULL)
m_list.m_description = m_list.add_string(attrvalues[1]);
}
else
unknown_tag(tagname);
}
//-------------------------------------------------
// parse_main_start - handle tag start within
// a softwarelist tag
//-------------------------------------------------
void softlist_parser::parse_main_start(const char *tagname, const char **attributes)
{
// <software name='' cloneof='' supported=''>
if (strcmp(tagname, "software") == 0)
{
static const char *attrnames[] = { "name", "cloneof", "supported" };
const char *attrvalues[ARRAY_LENGTH(attrnames)] = { 0 };
parse_attributes(attributes, ARRAY_LENGTH(attrnames), attrnames, attrvalues);
if (attrvalues[0] != NULL)
m_current_info = &m_list.m_infolist.append(*global_alloc(software_info(m_list, m_list.add_string(attrvalues[0]), m_list.add_string(attrvalues[1]), attrvalues[2])));
else
parse_error("No name defined for item");
}
else
unknown_tag(tagname);
}
//-------------------------------------------------
// parse_main_start - handle tag start within
// a software tag
//-------------------------------------------------
void softlist_parser::parse_soft_start(const char *tagname, const char **attributes)
{
// get the current info; error if none
if (m_current_info == NULL)
{
parse_error("Tag %s found outside of software context", tagname);
return;
}
// <description>
if (strcmp(tagname, "description") == 0)
m_data_accum_expected = true;
// <year>
else if (strcmp(tagname, "year") == 0)
m_data_accum_expected = true;
// <publisher>
else if (strcmp(tagname, "publisher") == 0)
m_data_accum_expected = true;
// <info name='' value=''>
else if (strcmp(tagname, "info") == 0)
{
static const char *attrnames[] = { "name", "value" };
const char *attrvalues[ARRAY_LENGTH(attrnames)] = { 0 };
parse_attributes(attributes, ARRAY_LENGTH(attrnames), attrnames, attrvalues);
if (attrvalues[0] != NULL && attrvalues[1] != NULL)
m_current_info->m_other_info.append(*global_alloc(feature_list_item(m_list.add_string(attrvalues[0]), m_list.add_string(attrvalues[1]))));
else
parse_error("Incomplete other_info definition");
}
// <sharedfeat name='' value=''>
else if (strcmp(tagname, "sharedfeat") == 0)
{
static const char *attrnames[] = { "name", "value" };
const char *attrvalues[ARRAY_LENGTH(attrnames)] = { 0 };
parse_attributes(attributes, ARRAY_LENGTH(attrnames), attrnames, attrvalues);
if (attrvalues[0] != NULL && attrvalues[1] != NULL)
m_current_info->m_shared_info.append(*global_alloc(feature_list_item(m_list.add_string(attrvalues[0]), m_list.add_string(attrvalues[1]))));
else
parse_error("Incomplete sharedfeat definition");
}
// <part name='' interface=''>
else if (strcmp(tagname, "part" ) == 0)
{
static const char *attrnames[] = { "name", "interface" };
const char *attrvalues[ARRAY_LENGTH(attrnames)] = { 0 };
parse_attributes(attributes, ARRAY_LENGTH(attrnames), attrnames, attrvalues);
if (attrvalues[0] != NULL && attrvalues[1] != NULL && strcmp(attrvalues[0], "") != 0 && strcmp(attrvalues[1], "") != 0)
m_current_part = &m_current_info->m_partdata.append(*global_alloc(software_part(*m_current_info, m_list.add_string(attrvalues[0]), m_list.add_string(attrvalues[1]))));
else
parse_error("Incomplete part definition");
}
else
unknown_tag(tagname);
}
//-------------------------------------------------
// parse_part_start - handle tag start within
// a part tag
//-------------------------------------------------
void softlist_parser::parse_part_start(const char *tagname, const char **attributes)
{
// get the current part; error if none
if (m_current_part == NULL)
{
parse_error("Tag %s found outside of part context", tagname);
return;
}
// <dataarea name='' size=''>
if (strcmp(tagname, "dataarea") == 0)
{
static const char *attrnames[] = { "name", "size", "width", "endianness" };
const char *attrvalues[ARRAY_LENGTH(attrnames)] = { 0 };
parse_attributes(attributes, ARRAY_LENGTH(attrnames), attrnames, attrvalues);
if (attrvalues[0] != NULL && attrvalues[1] != NULL && strcmp(attrvalues[0], "") != 0 && strcmp(attrvalues[1], "") != 0)
{
// handle region attributes
const char *width = attrvalues[2];
const char *endianness = attrvalues[3];
UINT32 regionflags = ROMENTRYTYPE_REGION;
if (width != NULL)
{
if (strcmp(width, "8") == 0)
regionflags |= ROMREGION_8BIT;
else if (strcmp(width, "16") == 0)
regionflags |= ROMREGION_16BIT;
else if (strcmp(width, "32") == 0)
regionflags |= ROMREGION_32BIT;
else if (strcmp(width, "64") == 0)
regionflags |= ROMREGION_64BIT;
else
parse_error("Invalid dataarea width");
}
if (endianness != NULL)
{
if (strcmp(endianness, "little") == 0)
regionflags |= ROMREGION_LE;
else if (strcmp(endianness, "big") == 0)
regionflags |= ROMREGION_BE;
else
parse_error("Invalid dataarea endianness");
}
add_rom_entry(attrvalues[0], NULL, 0, strtol(attrvalues[1], NULL, 0), regionflags);
}
else
parse_error("Incomplete dataarea definition");
}
// <diskarea name=''>
else if (strcmp(tagname, "diskarea") == 0)
{
static const char *attrnames[] = { "name" };
const char *attrvalues[ARRAY_LENGTH(attrnames)] = { 0 };
parse_attributes(attributes, ARRAY_LENGTH(attrnames), attrnames, attrvalues);
if (attrvalues[0] != NULL)
add_rom_entry(attrvalues[0], NULL, 0, 1, ROMENTRYTYPE_REGION | ROMREGION_DATATYPEDISK);
else
parse_error("Incomplete diskarea definition");
}
// <feature name='' value=''>
else if (strcmp(tagname, "feature") == 0)
{
static const char *attrnames[] = { "name", "value" };
const char *attrvalues[ARRAY_LENGTH(attrnames)] = { 0 };
parse_attributes(attributes, ARRAY_LENGTH(attrnames), attrnames, attrvalues);
if (attrvalues[0] != NULL)
m_current_part->m_featurelist.append(*global_alloc(feature_list_item(m_list.add_string(attrvalues[0]), m_list.add_string(attrvalues[1]))));
else
parse_error("Incomplete feature definition");
}
// <dipswitch>
else if (strcmp(tagname, "dipswitch") == 0)
;
else
unknown_tag(tagname);
}
//-------------------------------------------------
// parse_data_start - handle tag start within a
// dataarea or diskarea tag
//-------------------------------------------------
void softlist_parser::parse_data_start(const char *tagname, const char **attributes)
{
// get the current part; error if none
if (m_current_part == NULL)
{
parse_error("Tag %s found outside of part context", tagname);
return;
}
// <rom name='' size='' crc='' sha1='' offset='' value='' status='' loadflag=''>
if (strcmp(tagname, "rom") == 0)
{
static const char *attrnames[] = { "name", "size", "crc", "sha1", "offset", "value", "status", "loadflag" };
const char *attrvalues[ARRAY_LENGTH(attrnames)] = { 0 };
parse_attributes(attributes, ARRAY_LENGTH(attrnames), attrnames, attrvalues);
const char *name = attrvalues[0];
const char *sizestr = attrvalues[1];
const char *crc = attrvalues[2];
const char *sha1 = attrvalues[3];
const char *offsetstr = attrvalues[4];
const char *value = attrvalues[5];
const char *status = attrvalues[6];
const char *loadflag = attrvalues[7];
if (sizestr != NULL && offsetstr != NULL)
{
UINT32 length = strtol(sizestr, NULL, 0);
UINT32 offset = strtol(offsetstr, NULL, 0);
if (loadflag != NULL && strcmp(loadflag, "reload") == 0)
add_rom_entry(NULL, NULL, offset, length, ROMENTRYTYPE_RELOAD | ROM_INHERITFLAGS);
else if (loadflag != NULL && strcmp(loadflag, "reload_plain") == 0)
add_rom_entry(NULL, NULL, offset, length, ROMENTRYTYPE_RELOAD);
else if (loadflag != NULL && strcmp(loadflag, "continue") == 0)
add_rom_entry(NULL, NULL, offset, length, ROMENTRYTYPE_CONTINUE | ROM_INHERITFLAGS);
else if (loadflag != NULL && strcmp(loadflag, "fill") == 0)
add_rom_entry(NULL, (const char *)(FPTR)(strtol(value, NULL, 0) & 0xff), offset, length, ROMENTRYTYPE_FILL);
else if (name != NULL)
{
bool baddump = (status != NULL && strcmp(status, "baddump") == 0);
bool nodump = (status != NULL && strcmp(status, "nodump") == 0);
astring hashdata;
if (nodump)
{
hashdata.printf("%s", NO_DUMP);
if (crc != NULL && sha1 != NULL)
parse_error("No need for hash definition");
}
else
{
if (crc != NULL && sha1 != NULL)
hashdata.printf("%c%s%c%s%s", hash_collection::HASH_CRC, crc, hash_collection::HASH_SHA1, sha1, (baddump ? BAD_DUMP : ""));
else
parse_error("Incomplete rom hash definition");
}
// Handle loadflag attribute
int romflags = 0;
if (loadflag != NULL && strcmp(loadflag, "load16_word_swap") == 0)
romflags = ROM_GROUPWORD | ROM_REVERSE;
else if (loadflag != NULL && strcmp(loadflag, "load16_byte") == 0)
romflags = ROM_SKIP(1);
else if (loadflag != NULL && strcmp(loadflag, "load32_word_swap") == 0)
romflags = ROM_GROUPWORD | ROM_REVERSE | ROM_SKIP(2);
else if (loadflag != NULL && strcmp(loadflag, "load32_word") == 0)
romflags = ROM_GROUPWORD | ROM_SKIP(2);
else if (loadflag != NULL && strcmp(loadflag, "load32_byte") == 0)
romflags = ROM_SKIP(3);
add_rom_entry(name, hashdata.c_str(), offset, length, ROMENTRYTYPE_ROM | romflags);
}
else
parse_error("Rom name missing");
}
else
parse_error("Incomplete rom definition");
}
// <rom name='' sha1='' status='' writeable=''>
else if (strcmp(tagname, "disk") == 0)
{
static const char *attrnames[] = { "name", "sha1", "status", "writeable" };
const char *attrvalues[ARRAY_LENGTH(attrnames)] = { 0 };
parse_attributes(attributes, ARRAY_LENGTH(attrnames), attrnames, attrvalues);
const char *name = attrvalues[0];
const char *sha1 = attrvalues[1];
const char *status = attrvalues[2];
const char *writeablestr = attrvalues[3];
if (name != NULL && sha1 != NULL)
{
bool baddump = (status != NULL && strcmp(status, "baddump") == 0);
bool nodump = (status != NULL && strcmp(status, "nodump" ) == 0);
bool writeable = (writeablestr != NULL && strcmp(writeablestr, "yes") == 0);
astring hashdata;
hashdata.printf( "%c%s%s", hash_collection::HASH_SHA1, sha1, (nodump ? NO_DUMP : (baddump ? BAD_DUMP : "")));
add_rom_entry(name, hashdata.c_str(), 0, 0, ROMENTRYTYPE_ROM | (writeable ? DISK_READWRITE : DISK_READONLY));
}
else if (status == NULL || !strcmp(status, "nodump")) // a no_dump chd is not an incomplete entry
parse_error("Incomplete disk definition");
}
// <dipvalue>
else if (strcmp(tagname, "dipvalue") == 0)
;
else
unknown_tag(tagname);
}
//-------------------------------------------------
// parse_soft_end - handle end-of-tag post-
// processing within the <software> tag
//-------------------------------------------------
void softlist_parser::parse_soft_end(const char *tagname)
{
assert(m_current_info != NULL);
// <description>
if (strcmp(tagname, "description") == 0)
m_current_info->m_longname = m_list.add_string(m_data_accum.c_str());
// <year>
else if (strcmp(tagname, "year") == 0)
m_current_info->m_year = m_list.add_string(m_data_accum.c_str());
// <publisher>
else if (strcmp(tagname, "publisher") == 0)
m_current_info->m_publisher = m_list.add_string(m_data_accum.c_str());
// </part>
else if (strcmp(tagname, "part") == 0)
{
// get the last part
assert(m_current_part != NULL);
if (m_current_part == NULL)
return;
// was any dataarea/rom information encountered? if so, add a terminator
if (m_current_part->romdata() != NULL)
add_rom_entry(NULL, NULL, 0, 0, ROMENTRYTYPE_END);
// get the info; if present, copy shared data (we assume name/value strings live
// in the string pool and don't need to be reallocated)
if (m_current_info != NULL)
for (feature_list_item *item = m_current_info->shared_info(); item != NULL; item = item->next())
m_current_part->m_featurelist.append(*global_alloc(feature_list_item(item->name(), item->value())));
}
}