mame/src/emu/validity.c
Aaron Giles 4ea9df02a1 Moved core template container classes up from emutempl.h to coretmpl.h:
[Aaron Giles]
 * these classes now no longer take a resource_pool; everything is
    managed globally -- this means that objects added to lists must be
    allocated with global_alloc
 * added new auto_pointer<> template which wraps a pointer and auto-frees
    it upon destruction; it also defaults to NULL so it doesn't need to
    be explicitly initialized
 * moved tagged_list template to tagmap.h

Redo of the low-level memory tracking system: [Aaron Giles]
 * moved low-level tracking out of emu\emualloc into lib\util\corealloc
    so it can be shared among all components and used by core libraries
 * global_alloc and friends no longer use a resource pool to track
    allocations; turns out this was a wholly redundant system that wasted
    a lot of memory
 * removed global_resource_pool entirely
 * added global_free_array to delete arrays allocated with 
    global_alloc_array
 * added tracking of object versus array allocation; we will now error
    if you use global_free on an array, or global_free_array on an object

Added new utility helper const_string_pool which can be used to 
efficiently accumulate strings that are not intended to be modified.
Used by updated makelist and software list code. [Aaron Giles]

Updated png2bdc and makelist tools to not leak memory and use more modern
techniques (no more MAX_DRIVERS in makelist, for example). [Aaron Giles]

Deprecated auto_strdup and removed all uses by way of caller-managed 
astrings and the software list rewrite. [Aaron Giles]

Rewrote software list management: [Aaron Giles]
 * removed the notion of a software_list that is separate from a
    software_list_device; they are one and the same now
 * moved several functions into device_image_interface since they really
    didn't belong in the core software list class
 * lots of simplification as a result of the above changes

Additional notes (no whatsnew):

Moved definition of FPTR to osdcomm.h.

Some changes happened in the OSD code to fix issues, especially regarding
freeing arrays. SDL folks may need to fix up some of these.

The following devices still are using tokens and should be modernized
(I found them because they kept their token as void * and tried to
delete it, which you can't):

namco_52xx_device (mame/audio/namco52.c)
namco_54xx_device (mame/audio/namco54.c)
namco_06xx_device (mame/machine/namco06.c)
namco_50xx_device (mame/machine/namco50.c)
namco_51xx_device (mame/machine/namco51.c)
namco_53xx_device (mame/machine/namco53.c)
voodoo_device (emu/video/voodoo.c)
mos6581_device (emu/sound/mos6581.c)
aica_device (emu/sound/aica.c)
scsp_device (emu/sound/scsp.c)
dmadac_sound_device (emu/sound/dmadac.c)
s3c2440_device (emu/machine/s3c2440.c)
wd1770_device (emu/machine/wd17xx.c)
latch8_device (emu/machine/latch8.c)
duart68681_device (emu/machine/68681.c)
s3c2400_device (emu/machine/s3c2400.c)
s3c2410_device (emu/machine/s3c2410.c)
strataflash_device (mess/machine/strata.c)
hd63450_device (mess/machine/hd63450.c)
tap_990_device (mess/machine/ti99/990_tap.c)
omti8621_device (mess/machine/omti8621.c)
vdt911_device (mess/video/911_vdt.c)
apollo_graphics_15i (mess/video/apollo.c)
asr733_device (mess/video/733_asr.c)
2014-03-11 15:54:58 +00:00

1129 lines
40 KiB
C

// license:BSD-3-Clause
// copyright-holders:Aaron Giles
/***************************************************************************
validity.c
Validity checks on internal data structures.
***************************************************************************/
#include "emu.h"
#include "validity.h"
#include "emuopts.h"
#include <ctype.h>
//**************************************************************************
// COMPILE-TIME VALIDATION
//**************************************************************************
// if the following lines error during compile, your PTR64 switch is set incorrectly in the makefile
#ifdef PTR64
UINT8 your_ptr64_flag_is_wrong[(int)(sizeof(void *) - 7)];
#else
UINT8 your_ptr64_flag_is_wrong[(int)(5 - sizeof(void *))];
#endif
//**************************************************************************
// TYPE DEFINITIONS
//**************************************************************************
//**************************************************************************
// INLINE FUNCTIONS
//**************************************************************************
//-------------------------------------------------
// ioport_string_from_index - return an indexed
// string from the I/O port system
//-------------------------------------------------
inline const char *validity_checker::ioport_string_from_index(UINT32 index)
{
return ioport_configurer::string_from_token((const char *)(FPTR)index);
}
//-------------------------------------------------
// get_defstr_index - return the index of the
// string assuming it is one of the default
// strings
//-------------------------------------------------
inline int validity_checker::get_defstr_index(const char *string, bool suppress_error)
{
// check for strings that should be DEF_STR
int strindex = m_defstr_map.find(string);
if (!suppress_error && strindex != 0 && string != ioport_string_from_index(strindex))
mame_printf_error("Must use DEF_STR( %s )\n", string);
return strindex;
}
//-------------------------------------------------
// validate_tag - ensure that the given tag
// meets the general requirements
//-------------------------------------------------
void validity_checker::validate_tag(const char *tag)
{
// some common names that are now deprecated
if (strcmp(tag, "main") == 0 || strcmp(tag, "audio") == 0 || strcmp(tag, "sound") == 0 || strcmp(tag, "left") == 0 || strcmp(tag, "right") == 0)
mame_printf_error("Invalid generic tag '%s' used\n", tag);
// scan for invalid characters
static const char *validchars = "abcdefghijklmnopqrstuvwxyz0123456789_.:^$";
for (const char *p = tag; *p != 0; p++)
{
// only lower-case permitted
if (*p != tolower((UINT8)*p))
{
mame_printf_error("Tag '%s' contains upper-case characters\n", tag);
break;
}
if (*p == ' ')
{
mame_printf_error("Tag '%s' contains spaces\n", tag);
break;
}
if (strchr(validchars, *p) == NULL)
{
mame_printf_error("Tag '%s' contains invalid character '%c'\n", tag, *p);
break;
}
}
// find the start of the final tag
const char *begin = strrchr(tag, ':');
if (begin == NULL)
begin = tag;
else
begin += 1;
// 0-length = bad
if (*begin == 0)
mame_printf_error("Found 0-length tag\n");
// too short/too long = bad
if (strlen(begin) < MIN_TAG_LENGTH)
mame_printf_error("Tag '%s' is too short (must be at least %d characters)\n", tag, MIN_TAG_LENGTH);
if (strlen(begin) > MAX_TAG_LENGTH)
mame_printf_error("Tag '%s' is too long (must be less than %d characters)\n", tag, MAX_TAG_LENGTH);
}
//**************************************************************************
// VALIDATION FUNCTIONS
//**************************************************************************
//-------------------------------------------------
// validity_checker - constructor
//-------------------------------------------------
validity_checker::validity_checker(emu_options &options)
: m_drivlist(options),
m_errors(0),
m_warnings(0),
m_current_driver(NULL),
m_current_config(NULL),
m_current_device(NULL),
m_current_ioport(NULL)
{
// pre-populate the defstr map with all the default strings
for (int strnum = 1; strnum < INPUT_STRING_COUNT; strnum++)
{
const char *string = ioport_string_from_index(strnum);
if (string != NULL)
m_defstr_map.add(string, strnum, false);
}
}
//-------------------------------------------------
// check_driver - check a single driver
//-------------------------------------------------
void validity_checker::check_driver(const game_driver &driver)
{
// simply validate the one driver
validate_begin();
validate_one(driver);
validate_end();
}
//-------------------------------------------------
// check_shared_source - check all drivers that
// share the same source file as the given driver
//-------------------------------------------------
void validity_checker::check_shared_source(const game_driver &driver)
{
// initialize
validate_begin();
// then iterate over all drivers and check the ones that share the same source file
m_drivlist.reset();
while (m_drivlist.next())
if (strcmp(driver.source_file, m_drivlist.driver().source_file) == 0)
validate_one(m_drivlist.driver());
// cleanup
validate_end();
}
//-------------------------------------------------
// check_all - check all drivers
//-------------------------------------------------
void validity_checker::check_all()
{
// start by checking core stuff
validate_begin();
validate_core();
validate_inlines();
// if we had warnings or errors, output
if (m_errors > 0 || m_warnings > 0)
{
astring tempstr;
output_via_delegate(m_saved_error_output, "Core: %d errors, %d warnings\n", m_errors, m_warnings);
if (m_errors > 0)
{
m_error_text.replace("\n", "\n ");
output_via_delegate(m_saved_error_output, "Errors:\n %s", m_error_text.cstr());
}
if (m_warnings > 0)
{
m_warning_text.replace("\n", "\n ");
output_via_delegate(m_saved_error_output, "Warnings:\n %s", m_warning_text.cstr());
}
output_via_delegate(m_saved_error_output, "\n");
}
// then iterate over all drivers and check them
m_drivlist.reset();
while (m_drivlist.next())
validate_one(m_drivlist.driver());
// cleanup
validate_end();
}
//-------------------------------------------------
// validate_begin - prepare for validation by
// taking over the output callbacks and resetting
// our internal state
//-------------------------------------------------
void validity_checker::validate_begin()
{
// take over error and warning outputs
m_saved_error_output = mame_set_output_channel(OUTPUT_CHANNEL_ERROR, output_delegate(FUNC(validity_checker::error_output), this));
m_saved_warning_output = mame_set_output_channel(OUTPUT_CHANNEL_WARNING, output_delegate(FUNC(validity_checker::warning_output), this));
// reset all our maps
m_names_map.reset();
m_descriptions_map.reset();
m_roms_map.reset();
m_defstr_map.reset();
m_region_map.reset();
// reset internal state
m_errors = 0;
m_warnings = 0;
m_already_checked.reset();
}
//-------------------------------------------------
// validate_end - restore output callbacks and
// clean up
//-------------------------------------------------
void validity_checker::validate_end()
{
// restore the original output callbacks
mame_set_output_channel(OUTPUT_CHANNEL_ERROR, m_saved_error_output);
mame_set_output_channel(OUTPUT_CHANNEL_WARNING, m_saved_warning_output);
}
//-------------------------------------------------
// validate_drivers - master validity checker
//-------------------------------------------------
void validity_checker::validate_one(const game_driver &driver)
{
// set the current driver
m_current_driver = &driver;
m_current_config = NULL;
m_current_device = NULL;
m_current_ioport = NULL;
m_region_map.reset();
// reset error/warning state
int start_errors = m_errors;
int start_warnings = m_warnings;
m_error_text.reset();
m_warning_text.reset();
// wrap in try/except to catch fatalerrors
try
{
machine_config config(driver, m_drivlist.options());
m_current_config = &config;
validate_driver();
validate_roms();
validate_inputs();
validate_display();
validate_devices();
}
catch (emu_fatalerror &err)
{
mame_printf_error("Fatal error %s", err.string());
}
m_current_config = NULL;
// if we had warnings or errors, output
if (m_errors > start_errors || m_warnings > start_warnings)
{
astring tempstr;
output_via_delegate(m_saved_error_output, "Driver %s (file %s): %d errors, %d warnings\n", driver.name, core_filename_extract_base(tempstr, driver.source_file).cstr(), m_errors - start_errors, m_warnings - start_warnings);
if (m_errors > start_errors)
{
m_error_text.replace("\n", "\n ");
output_via_delegate(m_saved_error_output, "Errors:\n %s", m_error_text.cstr());
}
if (m_warnings > start_warnings)
{
m_warning_text.replace("\n", "\n ");
output_via_delegate(m_saved_error_output, "Warnings:\n %s", m_warning_text.cstr());
}
output_via_delegate(m_saved_error_output, "\n");
}
// reset the driver/device
m_current_driver = NULL;
m_current_config = NULL;
m_current_device = NULL;
m_current_ioport = NULL;
}
//-------------------------------------------------
// validate_core - validate core internal systems
//-------------------------------------------------
void validity_checker::validate_core()
{
// basic system checks
UINT8 a = 0xff;
UINT8 b = a + 1;
if (b > a) mame_printf_error("UINT8 must be 8 bits\n");
// check size of core integer types
if (sizeof(INT8) != 1) mame_printf_error("INT8 must be 8 bits\n");
if (sizeof(UINT8) != 1) mame_printf_error("UINT8 must be 8 bits\n");
if (sizeof(INT16) != 2) mame_printf_error("INT16 must be 16 bits\n");
if (sizeof(UINT16) != 2) mame_printf_error("UINT16 must be 16 bits\n");
if (sizeof(INT32) != 4) mame_printf_error("INT32 must be 32 bits\n");
if (sizeof(UINT32) != 4) mame_printf_error("UINT32 must be 32 bits\n");
if (sizeof(INT64) != 8) mame_printf_error("INT64 must be 64 bits\n");
if (sizeof(UINT64) != 8) mame_printf_error("UINT64 must be 64 bits\n");
// check pointer size
#ifdef PTR64
if (sizeof(void *) != 8) mame_printf_error("PTR64 flag enabled, but was compiled for 32-bit target\n");
#else
if (sizeof(void *) != 4) mame_printf_error("PTR64 flag not enabled, but was compiled for 64-bit target\n");
#endif
// check endianness definition
UINT16 lsbtest = 0;
*(UINT8 *)&lsbtest = 0xff;
#ifdef LSB_FIRST
if (lsbtest == 0xff00) mame_printf_error("LSB_FIRST specified, but running on a big-endian machine\n");
#else
if (lsbtest == 0x00ff) mame_printf_error("LSB_FIRST not specified, but running on a little-endian machine\n");
#endif
}
//-------------------------------------------------
// validate_inlines - validate inline function
// behaviors
//-------------------------------------------------
void validity_checker::validate_inlines()
{
#undef rand
volatile UINT64 testu64a = rand() ^ (rand() << 15) ^ ((UINT64)rand() << 30) ^ ((UINT64)rand() << 45);
volatile INT64 testi64a = rand() ^ (rand() << 15) ^ ((INT64)rand() << 30) ^ ((INT64)rand() << 45);
#ifdef PTR64
volatile INT64 testi64b = rand() ^ (rand() << 15) ^ ((INT64)rand() << 30) ^ ((INT64)rand() << 45);
#endif
volatile UINT32 testu32a = rand() ^ (rand() << 15);
volatile UINT32 testu32b = rand() ^ (rand() << 15);
volatile INT32 testi32a = rand() ^ (rand() << 15);
volatile INT32 testi32b = rand() ^ (rand() << 15);
INT32 resulti32, expectedi32;
UINT32 resultu32, expectedu32;
INT64 resulti64, expectedi64;
UINT64 resultu64, expectedu64;
INT32 remainder, expremainder;
UINT32 uremainder, expuremainder, bigu32 = 0xffffffff;
// use only non-zero, positive numbers
if (testu64a == 0) testu64a++;
if (testi64a == 0) testi64a++;
else if (testi64a < 0) testi64a = -testi64a;
#ifdef PTR64
if (testi64b == 0) testi64b++;
else if (testi64b < 0) testi64b = -testi64b;
#endif
if (testu32a == 0) testu32a++;
if (testu32b == 0) testu32b++;
if (testi32a == 0) testi32a++;
else if (testi32a < 0) testi32a = -testi32a;
if (testi32b == 0) testi32b++;
else if (testi32b < 0) testi32b = -testi32b;
resulti64 = mul_32x32(testi32a, testi32b);
expectedi64 = (INT64)testi32a * (INT64)testi32b;
if (resulti64 != expectedi64)
mame_printf_error("Error testing mul_32x32 (%08X x %08X) = %08X%08X (expected %08X%08X)\n", testi32a, testi32b, (UINT32)(resulti64 >> 32), (UINT32)resulti64, (UINT32)(expectedi64 >> 32), (UINT32)expectedi64);
resultu64 = mulu_32x32(testu32a, testu32b);
expectedu64 = (UINT64)testu32a * (UINT64)testu32b;
if (resultu64 != expectedu64)
mame_printf_error("Error testing mulu_32x32 (%08X x %08X) = %08X%08X (expected %08X%08X)\n", testu32a, testu32b, (UINT32)(resultu64 >> 32), (UINT32)resultu64, (UINT32)(expectedu64 >> 32), (UINT32)expectedu64);
resulti32 = mul_32x32_hi(testi32a, testi32b);
expectedi32 = ((INT64)testi32a * (INT64)testi32b) >> 32;
if (resulti32 != expectedi32)
mame_printf_error("Error testing mul_32x32_hi (%08X x %08X) = %08X (expected %08X)\n", testi32a, testi32b, resulti32, expectedi32);
resultu32 = mulu_32x32_hi(testu32a, testu32b);
expectedu32 = ((INT64)testu32a * (INT64)testu32b) >> 32;
if (resultu32 != expectedu32)
mame_printf_error("Error testing mulu_32x32_hi (%08X x %08X) = %08X (expected %08X)\n", testu32a, testu32b, resultu32, expectedu32);
resulti32 = mul_32x32_shift(testi32a, testi32b, 7);
expectedi32 = ((INT64)testi32a * (INT64)testi32b) >> 7;
if (resulti32 != expectedi32)
mame_printf_error("Error testing mul_32x32_shift (%08X x %08X) >> 7 = %08X (expected %08X)\n", testi32a, testi32b, resulti32, expectedi32);
resultu32 = mulu_32x32_shift(testu32a, testu32b, 7);
expectedu32 = ((INT64)testu32a * (INT64)testu32b) >> 7;
if (resultu32 != expectedu32)
mame_printf_error("Error testing mulu_32x32_shift (%08X x %08X) >> 7 = %08X (expected %08X)\n", testu32a, testu32b, resultu32, expectedu32);
while ((INT64)testi32a * (INT64)0x7fffffff < testi64a)
testi64a /= 2;
while ((UINT64)testu32a * (UINT64)bigu32 < testu64a)
testu64a /= 2;
resulti32 = div_64x32(testi64a, testi32a);
expectedi32 = testi64a / (INT64)testi32a;
if (resulti32 != expectedi32)
mame_printf_error("Error testing div_64x32 (%08X%08X / %08X) = %08X (expected %08X)\n", (UINT32)(testi64a >> 32), (UINT32)testi64a, testi32a, resulti32, expectedi32);
resultu32 = divu_64x32(testu64a, testu32a);
expectedu32 = testu64a / (UINT64)testu32a;
if (resultu32 != expectedu32)
mame_printf_error("Error testing divu_64x32 (%08X%08X / %08X) = %08X (expected %08X)\n", (UINT32)(testu64a >> 32), (UINT32)testu64a, testu32a, resultu32, expectedu32);
resulti32 = div_64x32_rem(testi64a, testi32a, &remainder);
expectedi32 = testi64a / (INT64)testi32a;
expremainder = testi64a % (INT64)testi32a;
if (resulti32 != expectedi32 || remainder != expremainder)
mame_printf_error("Error testing div_64x32_rem (%08X%08X / %08X) = %08X,%08X (expected %08X,%08X)\n", (UINT32)(testi64a >> 32), (UINT32)testi64a, testi32a, resulti32, remainder, expectedi32, expremainder);
resultu32 = divu_64x32_rem(testu64a, testu32a, &uremainder);
expectedu32 = testu64a / (UINT64)testu32a;
expuremainder = testu64a % (UINT64)testu32a;
if (resultu32 != expectedu32 || uremainder != expuremainder)
mame_printf_error("Error testing divu_64x32_rem (%08X%08X / %08X) = %08X,%08X (expected %08X,%08X)\n", (UINT32)(testu64a >> 32), (UINT32)testu64a, testu32a, resultu32, uremainder, expectedu32, expuremainder);
resulti32 = mod_64x32(testi64a, testi32a);
expectedi32 = testi64a % (INT64)testi32a;
if (resulti32 != expectedi32)
mame_printf_error("Error testing mod_64x32 (%08X%08X / %08X) = %08X (expected %08X)\n", (UINT32)(testi64a >> 32), (UINT32)testi64a, testi32a, resulti32, expectedi32);
resultu32 = modu_64x32(testu64a, testu32a);
expectedu32 = testu64a % (UINT64)testu32a;
if (resultu32 != expectedu32)
mame_printf_error("Error testing modu_64x32 (%08X%08X / %08X) = %08X (expected %08X)\n", (UINT32)(testu64a >> 32), (UINT32)testu64a, testu32a, resultu32, expectedu32);
while ((INT64)testi32a * (INT64)0x7fffffff < ((INT32)testi64a << 3))
testi64a /= 2;
while ((UINT64)testu32a * (UINT64)0xffffffff < ((UINT32)testu64a << 3))
testu64a /= 2;
resulti32 = div_32x32_shift((INT32)testi64a, testi32a, 3);
expectedi32 = ((INT64)(INT32)testi64a << 3) / (INT64)testi32a;
if (resulti32 != expectedi32)
mame_printf_error("Error testing div_32x32_shift (%08X << 3) / %08X = %08X (expected %08X)\n", (INT32)testi64a, testi32a, resulti32, expectedi32);
resultu32 = divu_32x32_shift((UINT32)testu64a, testu32a, 3);
expectedu32 = ((UINT64)(UINT32)testu64a << 3) / (UINT64)testu32a;
if (resultu32 != expectedu32)
mame_printf_error("Error testing divu_32x32_shift (%08X << 3) / %08X = %08X (expected %08X)\n", (UINT32)testu64a, testu32a, resultu32, expectedu32);
if (fabs(recip_approx(100.0) - 0.01) > 0.0001)
mame_printf_error("Error testing recip_approx\n");
testi32a = (testi32a & 0x0000ffff) | 0x400000;
if (count_leading_zeros(testi32a) != 9)
mame_printf_error("Error testing count_leading_zeros\n");
testi32a = (testi32a | 0xffff0000) & ~0x400000;
if (count_leading_ones(testi32a) != 9)
mame_printf_error("Error testing count_leading_ones\n");
testi32b = testi32a;
if (compare_exchange32(&testi32a, testi32b, 1000) != testi32b || testi32a != 1000)
mame_printf_error("Error testing compare_exchange32\n");
#ifdef PTR64
testi64b = testi64a;
if (compare_exchange64(&testi64a, testi64b, 1000) != testi64b || testi64a != 1000)
mame_printf_error("Error testing compare_exchange64\n");
#endif
if (atomic_exchange32(&testi32a, testi32b) != 1000)
mame_printf_error("Error testing atomic_exchange32\n");
if (atomic_add32(&testi32a, 45) != testi32b + 45)
mame_printf_error("Error testing atomic_add32\n");
if (atomic_increment32(&testi32a) != testi32b + 46)
mame_printf_error("Error testing atomic_increment32\n");
if (atomic_decrement32(&testi32a) != testi32b + 45)
mame_printf_error("Error testing atomic_decrement32\n");
}
//-------------------------------------------------
// validate_driver - validate basic driver
// information
//-------------------------------------------------
void validity_checker::validate_driver()
{
// check for duplicate names
astring tempstr;
if (m_names_map.add(m_current_driver->name, m_current_driver, false) == TMERR_DUPLICATE)
{
const game_driver *match = m_names_map.find(m_current_driver->name);
mame_printf_error("Driver name is a duplicate of %s(%s)\n", core_filename_extract_base(tempstr, match->source_file).cstr(), match->name);
}
// check for duplicate descriptions
if (m_descriptions_map.add(m_current_driver->description, m_current_driver, false) == TMERR_DUPLICATE)
{
const game_driver *match = m_descriptions_map.find(m_current_driver->description);
mame_printf_error("Driver description is a duplicate of %s(%s)\n", core_filename_extract_base(tempstr, match->source_file).cstr(), match->name);
}
// determine if we are a clone
bool is_clone = (strcmp(m_current_driver->parent, "0") != 0);
int clone_of = m_drivlist.clone(*m_current_driver);
if (clone_of != -1 && (m_drivlist.driver(clone_of).flags & GAME_IS_BIOS_ROOT))
is_clone = false;
// if we have at least 100 drivers, validate the clone
// (100 is arbitrary, but tries to avoid tiny.mak dependencies)
if (driver_list::total() > 100 && clone_of == -1 && is_clone)
mame_printf_error("Driver is a clone of nonexistant driver %s\n", m_current_driver->parent);
// look for recursive cloning
if (clone_of != -1 && &m_drivlist.driver(clone_of) == m_current_driver)
mame_printf_error("Driver is a clone of itself\n");
// look for clones that are too deep
if (clone_of != -1 && (clone_of = m_drivlist.non_bios_clone(clone_of)) != -1)
mame_printf_error("Driver is a clone of a clone\n");
// make sure the driver name is not too long
if (!is_clone && strlen(m_current_driver->name) > 8)
mame_printf_error("Parent driver name must be 8 characters or less\n");
if (is_clone && strlen(m_current_driver->name) > 16)
mame_printf_error("Clone driver name must be 16 characters or less\n");
// make sure the year is only digits, '?' or '+'
for (const char *s = m_current_driver->year; *s != 0; s++)
if (!isdigit((UINT8)*s) && *s != '?' && *s != '+')
{
mame_printf_error("Driver has an invalid year '%s'\n", m_current_driver->year);
break;
}
// normalize driver->compatible_with
const char *compatible_with = m_current_driver->compatible_with;
if (compatible_with != NULL && strcmp(compatible_with, "0") == 0)
compatible_with = NULL;
// check for this driver being compatible with a non-existant driver
if (compatible_with != NULL && m_drivlist.find(m_current_driver->compatible_with) == -1)
mame_printf_error("Driver is listed as compatible with nonexistant driver %s\n", m_current_driver->compatible_with);
// check for clone_of and compatible_with being specified at the same time
if (m_drivlist.clone(*m_current_driver) != -1 && compatible_with != NULL)
mame_printf_error("Driver cannot be both a clone and listed as compatible with another system\n");
// find any recursive dependencies on the current driver
for (int other_drv = m_drivlist.compatible_with(*m_current_driver); other_drv != -1; other_drv = m_drivlist.compatible_with(other_drv))
if (m_current_driver == &m_drivlist.driver(other_drv))
{
mame_printf_error("Driver is recursively compatible with itself\n");
break;
}
// make sure sound-less drivers are flagged
sound_interface_iterator iter(m_current_config->root_device());
if ((m_current_driver->flags & GAME_IS_BIOS_ROOT) == 0 && iter.first() == NULL && (m_current_driver->flags & GAME_NO_SOUND) == 0 && (m_current_driver->flags & GAME_NO_SOUND_HW) == 0)
mame_printf_error("Driver is missing GAME_NO_SOUND flag\n");
}
//-------------------------------------------------
// validate_roms - validate ROM definitions
//-------------------------------------------------
void validity_checker::validate_roms()
{
// iterate, starting with the driver's ROMs and continuing with device ROMs
device_iterator deviter(m_current_config->root_device());
for (device_t *device = deviter.first(); device != NULL; device = deviter.next())
{
// for non-root devices, track the current device
m_current_device = (device->owner() == NULL) ? NULL : device;
// scan the ROM entries for this device
const char *last_region_name = "???";
const char *last_name = "???";
UINT32 current_length = 0;
int items_since_region = 1;
int last_bios = 0;
int total_files = 0;
for (const rom_entry *romp = rom_first_region(*device); romp != NULL && !ROMENTRY_ISEND(romp); romp++)
{
// if this is a region, make sure it's valid, and record the length
if (ROMENTRY_ISREGION(romp))
{
// if we haven't seen any items since the last region, print a warning
if (items_since_region == 0)
mame_printf_warning("Empty ROM region '%s' (warning)\n", last_region_name);
// reset our region tracking states
const char *basetag = ROMREGION_GETTAG(romp);
items_since_region = (ROMREGION_ISERASE(romp) || ROMREGION_ISDISKDATA(romp)) ? 1 : 0;
last_region_name = basetag;
// check for a valid tag
if (basetag == NULL)
{
mame_printf_error("ROM_REGION tag with NULL name\n");
continue;
}
// validate the base tag
validate_tag(basetag);
// generate the full tag
astring fulltag;
rom_region_name(fulltag, *device, romp);
// attempt to add it to the map, reporting duplicates as errors
current_length = ROMREGION_GETLENGTH(romp);
if (m_region_map.add(fulltag, current_length, false) == TMERR_DUPLICATE)
mame_printf_error("Multiple ROM_REGIONs with the same tag '%s' defined\n", fulltag.cstr());
}
// If this is a system bios, make sure it is using the next available bios number
else if (ROMENTRY_ISSYSTEM_BIOS(romp))
{
int bios_flags = ROM_GETBIOSFLAGS(romp);
if (bios_flags != last_bios + 1)
mame_printf_error("Non-sequential bios %s (specified as %d, expected to be %d)\n", ROM_GETNAME(romp), bios_flags, last_bios + 1);
last_bios = bios_flags;
}
// if this is a file, make sure it is properly formatted
else if (ROMENTRY_ISFILE(romp))
{
// track the last filename we found
last_name = ROM_GETNAME(romp);
total_files++;
// make sure it's all lowercase
for (const char *s = last_name; *s != 0; s++)
if (tolower((UINT8)*s) != *s)
{
mame_printf_error("ROM name '%s' contains upper case characters\n", last_name);
break;
}
// make sure the hash is valid
hash_collection hashes;
if (!hashes.from_internal_string(ROM_GETHASHDATA(romp)))
mame_printf_error("ROM '%s' has an invalid hash string '%s'\n", last_name, ROM_GETHASHDATA(romp));
}
// for any non-region ending entries, make sure they don't extend past the end
if (!ROMENTRY_ISREGIONEND(romp) && current_length > 0)
{
items_since_region++;
if (ROM_GETOFFSET(romp) + ROM_GETLENGTH(romp) > current_length)
mame_printf_error("ROM '%s' extends past the defined memory region\n", last_name);
}
}
// final check for empty regions
if (items_since_region == 0)
mame_printf_warning("Empty ROM region '%s' (warning)\n", last_region_name);
// reset the current device
m_current_device = NULL;
}
}
//-------------------------------------------------
// validate_display - validate display
// configurations
//-------------------------------------------------
void validity_checker::validate_display()
{
// iterate over screen devices looking for paletted screens
screen_device_iterator iter(m_current_config->root_device());
for (const screen_device *scrconfig = iter.first(); scrconfig != NULL; scrconfig = iter.next())
if (scrconfig->format() == BITMAP_FORMAT_IND16)
{
// check for empty palette
//if (scrconfig->palette()->entries() == 0)
// mame_printf_error("Driver has zero palette entries but uses a palette-based bitmap format\n");
}
}
//-------------------------------------------------
// validate_analog_input_field - validate an
// analog input field
//-------------------------------------------------
void validity_checker::validate_analog_input_field(ioport_field &field)
{
// analog ports must have a valid sensitivity
if (field.sensitivity() == 0)
mame_printf_error("Analog port with zero sensitivity\n");
// check that the default falls in the bitmask range
if (field.defvalue() & ~field.mask())
mame_printf_error("Analog port with a default value (%X) out of the bitmask range (%X)\n", field.defvalue(), field.mask());
// tests for positional devices
if (field.type() == IPT_POSITIONAL || field.type() == IPT_POSITIONAL_V)
{
int shift;
for (shift = 0; shift <= 31 && (~field.mask() & (1 << shift)) != 0; shift++) ;
// convert the positional max value to be in the bitmask for testing
INT32 analog_max = field.maxval();
analog_max = (analog_max - 1) << shift;
// positional port size must fit in bits used
if ((field.mask() >> shift) + 1 < field.maxval())
mame_printf_error("Analog port with a positional port size bigger then the mask size\n");
}
// tests for absolute devices
else if (field.type() > IPT_ANALOG_ABSOLUTE_FIRST && field.type() < IPT_ANALOG_ABSOLUTE_LAST)
{
// adjust for signed values
INT32 default_value = field.defvalue();
INT32 analog_min = field.minval();
INT32 analog_max = field.maxval();
if (analog_min > analog_max)
{
analog_min = -analog_min;
if (default_value > analog_max)
default_value = -default_value;
}
// check that the default falls in the MINMAX range
if (default_value < analog_min || default_value > analog_max)
mame_printf_error("Analog port with a default value (%X) out of PORT_MINMAX range (%X-%X)\n", field.defvalue(), field.minval(), field.maxval());
// check that the MINMAX falls in the bitmask range
// we use the unadjusted min for testing
if (field.minval() & ~field.mask() || analog_max & ~field.mask())
mame_printf_error("Analog port with a PORT_MINMAX (%X-%X) value out of the bitmask range (%X)\n", field.minval(), field.maxval(), field.mask());
// absolute analog ports do not use PORT_RESET
if (field.analog_reset())
mame_printf_error("Absolute analog port using PORT_RESET\n");
// absolute analog ports do not use PORT_WRAPS
if (field.analog_wraps())
mame_printf_error("Absolute analog port using PORT_WRAPS\n");
}
// tests for non IPT_POSITIONAL relative devices
else
{
// relative devices do not use PORT_MINMAX
if (field.minval() != 0 || field.maxval() != field.mask())
mame_printf_error("Relative port using PORT_MINMAX\n");
// relative devices do not use a default value
// the counter is at 0 on power up
if (field.defvalue() != 0)
mame_printf_error("Relative port using non-0 default value\n");
// relative analog ports do not use PORT_WRAPS
if (field.analog_wraps())
mame_printf_error("Absolute analog port using PORT_WRAPS\n");
}
}
//-------------------------------------------------
// validate_dip_settings - validate a DIP switch
// setting
//-------------------------------------------------
void validity_checker::validate_dip_settings(ioport_field &field)
{
const char *demo_sounds = ioport_string_from_index(INPUT_STRING_Demo_Sounds);
const char *flipscreen = ioport_string_from_index(INPUT_STRING_Flip_Screen);
UINT8 coin_list[__input_string_coinage_end + 1 - __input_string_coinage_start] = { 0 };
bool coin_error = false;
// iterate through the settings
for (ioport_setting *setting = field.first_setting(); setting != NULL; setting = setting->next())
{
// note any coinage strings
int strindex = get_defstr_index(setting->name());
if (strindex >= __input_string_coinage_start && strindex <= __input_string_coinage_end)
coin_list[strindex - __input_string_coinage_start] = 1;
// make sure demo sounds default to on
if (field.name() == demo_sounds && strindex == INPUT_STRING_On && field.defvalue() != setting->value())
mame_printf_error("Demo Sounds must default to On\n");
// check for bad demo sounds options
if (field.name() == demo_sounds && (strindex == INPUT_STRING_Yes || strindex == INPUT_STRING_No))
mame_printf_error("Demo Sounds option must be Off/On, not %s\n", setting->name());
// check for bad flip screen options
if (field.name() == flipscreen && (strindex == INPUT_STRING_Yes || strindex == INPUT_STRING_No))
mame_printf_error("Flip Screen option must be Off/On, not %s\n", setting->name());
// if we have a neighbor, compare ourselves to him
if (setting->next() != NULL)
{
// check for inverted off/on dispswitch order
int next_strindex = get_defstr_index(setting->next()->name(), true);
if (strindex == INPUT_STRING_On && next_strindex == INPUT_STRING_Off)
mame_printf_error("%s option must have Off/On options in the order: Off, On\n", field.name());
// check for inverted yes/no dispswitch order
else if (strindex == INPUT_STRING_Yes && next_strindex == INPUT_STRING_No)
mame_printf_error("%s option must have Yes/No options in the order: No, Yes\n", field.name());
// check for inverted upright/cocktail dispswitch order
else if (strindex == INPUT_STRING_Cocktail && next_strindex == INPUT_STRING_Upright)
mame_printf_error("%s option must have Upright/Cocktail options in the order: Upright, Cocktail\n", field.name());
// check for proper coin ordering
else if (strindex >= __input_string_coinage_start && strindex <= __input_string_coinage_end && next_strindex >= __input_string_coinage_start && next_strindex <= __input_string_coinage_end &&
strindex >= next_strindex && setting->condition() == setting->next()->condition())
{
mame_printf_error("%s option has unsorted coinage %s > %s\n", field.name(), setting->name(), setting->next()->name());
coin_error = true;
}
}
}
// if we have a coin error, demonstrate the correct way
if (coin_error)
{
output_via_delegate(m_saved_error_output, " Note proper coin sort order should be:\n");
for (int entry = 0; entry < ARRAY_LENGTH(coin_list); entry++)
if (coin_list[entry])
output_via_delegate(m_saved_error_output, " %s\n", ioport_string_from_index(__input_string_coinage_start + entry));
}
}
//-------------------------------------------------
// validate_condition - validate a condition
// stored within an ioport field or setting
//-------------------------------------------------
void validity_checker::validate_condition(ioport_condition &condition, device_t &device, int_map &port_map)
{
// resolve the tag
astring porttag;
device.subtag(porttag, condition.tag());
// then find a matching port
if (port_map.find(porttag) == 0)
mame_printf_error("Condition referencing non-existent ioport tag '%s'\n", condition.tag());
}
//-------------------------------------------------
// validate_inputs - validate input configuration
//-------------------------------------------------
void validity_checker::validate_inputs()
{
int_map port_map;
// iterate over devices
device_iterator iter(m_current_config->root_device());
for (device_t *device = iter.first(); device != NULL; device = iter.next())
{
// see if this device has ports; if not continue
if (device->input_ports() == NULL)
continue;
// for non-root devices, track the current device
m_current_device = (device == &m_current_config->root_device()) ? NULL : device;
// allocate the input ports
ioport_list portlist;
astring errorbuf;
portlist.append(*device, errorbuf);
// report any errors during construction
if (errorbuf)
mame_printf_error("I/O port error during construction:\n%s\n", errorbuf.cstr());
// do a first pass over ports to add their names and find duplicates
for (ioport_port *port = portlist.first(); port != NULL; port = port->next())
if (port_map.add(port->tag(), 1, false) == TMERR_DUPLICATE)
mame_printf_error("Multiple I/O ports with the same tag '%s' defined\n", port->tag());
// iterate over ports
for (ioport_port *port = portlist.first(); port != NULL; port = port->next())
{
m_current_ioport = port->tag();
// iterate through the fields on this port
for (ioport_field *field = port->first_field(); field != NULL; field = field->next())
{
// verify analog inputs
if (field->is_analog())
validate_analog_input_field(*field);
// look for invalid (0) types which should be mapped to IPT_OTHER
if (field->type() == IPT_INVALID)
mame_printf_error("Field has an invalid type (0); use IPT_OTHER instead\n");
// verify dip switches
if (field->type() == IPT_DIPSWITCH)
{
// dip switch fields must have a name
if (field->name() == NULL)
mame_printf_error("DIP switch has a NULL name\n");
// verify the settings list
validate_dip_settings(*field);
}
// verify names
const char *name = field->specific_name();
if (name != NULL)
{
// check for empty string
if (name[0] == 0)
mame_printf_error("Field name is an empty string\n");
// check for trailing spaces
if (name[0] != 0 && name[strlen(name) - 1] == ' ')
mame_printf_error("Field '%s' has trailing spaces\n", name);
// check for invalid UTF-8
if (!utf8_is_valid_string(name))
mame_printf_error("Field '%s' has invalid characters\n", name);
// look up the string and print an error if default strings are not used
/*strindex =get_defstr_index(defstr_map, name, driver, &error);*/
}
// verify conditions on the field
if (!field->condition().none())
validate_condition(field->condition(), *device, port_map);
// verify conditions on the settings
for (ioport_setting *setting = field->first_setting(); setting != NULL; setting = setting->next())
if (!setting->condition().none())
validate_condition(setting->condition(), *device, port_map);
}
// done with this port
m_current_ioport = NULL;
}
// done with this device
m_current_device = NULL;
}
}
//-------------------------------------------------
// validate_devices - run per-device validity
// checks
//-------------------------------------------------
void validity_checker::validate_devices()
{
int_map device_map;
device_iterator iter_find(m_current_config->root_device());
for (const device_t *device = iter_find.first(); device != NULL; device = iter_find.next())
{
device->findit(true);
}
// iterate over devices
device_iterator iter(m_current_config->root_device());
for (const device_t *device = iter.first(); device != NULL; device = iter.next())
{
// for non-root devices, track the current device
m_current_device = (device == &m_current_config->root_device()) ? NULL : device;
// validate the device tag
validate_tag(device->basetag());
// look for duplicates
if (device_map.add(device->tag(), 0, false) == TMERR_DUPLICATE)
mame_printf_error("Multiple devices with the same tag '%s' defined\n", device->tag());
// all devices must have a shortname
if (strcmp(device->shortname(), "") == 0)
mame_printf_error("Device does not have short name defined\n");
// all devices must have a source file defined
if (strcmp(device->source(), "") == 0)
mame_printf_error("Device does not have source file location defined\n");
// check for device-specific validity check
device->validity_check(*this);
// done with this device
m_current_device = NULL;
}
// if device is slot cart device, we must have a shortname
int_map slot_device_map;
slot_interface_iterator slotiter(m_current_config->root_device());
for (const device_slot_interface *slot = slotiter.first(); slot != NULL; slot = slotiter.next())
{
for (const device_slot_option *option = slot->first_option(); option != NULL; option = option->next())
{
astring temptag("_");
temptag.cat(option->name());
device_t *dev = const_cast<machine_config &>(*m_current_config).device_add(&m_current_config->root_device(), temptag.cstr(), option->devtype(), 0);
// notify this device and all its subdevices that they are now configured
device_iterator subiter(*dev);
for (device_t *device = subiter.first(); device != NULL; device = subiter.next())
if (!device->configured())
device->config_complete();
if (strcmp(dev->shortname(), "") == 0) {
if (slot_device_map.add(dev->name(), 0, false) != TMERR_DUPLICATE)
mame_printf_error("Device '%s' is slot cart device but does not have short name defined\n",dev->name());
}
const_cast<machine_config &>(*m_current_config).device_remove(&m_current_config->root_device(), temptag.cstr());
}
}
}
//-------------------------------------------------
// build_output_prefix - create a prefix
// indicating the current source file, driver,
// and device
//-------------------------------------------------
void validity_checker::build_output_prefix(astring &string)
{
// start empty
string.reset();
// if we have a current device, indicate that
if (m_current_device != NULL)
string.cat(m_current_device->name()).cat(" device '").cat(m_current_device->tag()).cat("': ");
// if we have a current port, indicate that as well
if (m_current_ioport != NULL)
string.cat("ioport '").cat(m_current_ioport).cat("': ");
}
//-------------------------------------------------
// error_output - error message output override
//-------------------------------------------------
void validity_checker::error_output(const char *format, va_list argptr)
{
// count the error
m_errors++;
// output the source(driver) device 'tag'
astring output;
build_output_prefix(output);
// generate the string
output.catvprintf(format, argptr);
m_error_text.cat(output);
}
//-------------------------------------------------
// warning_output - warning message output
// override
//-------------------------------------------------
void validity_checker::warning_output(const char *format, va_list argptr)
{
// count the error
m_warnings++;
// output the source(driver) device 'tag'
astring output;
build_output_prefix(output);
// generate the string and output to the original target
output.catvprintf(format, argptr);
m_warning_text.cat(output);
}
//-------------------------------------------------
// output_via_delegate - helper to output a
// message via a varargs string, so the argptr
// can be forwarded onto the given delegate
//-------------------------------------------------
void validity_checker::output_via_delegate(output_delegate &delegate, const char *format, ...)
{
va_list argptr;
// call through to the delegate with the proper parameters
va_start(argptr, format);
delegate(format, argptr);
va_end(argptr);
}