mirror of
https://github.com/holub/mame
synced 2025-10-04 16:34:53 +03:00

from 4 to 5. This means any diff CHDs will no longer work. If you absolutely need to keep the data for any existing ones you have, find both the diff CHD and the original CHD for the game in question and upgrade using these commands: rename diff\game.dif diff\game-old.dif chdman copy -i diff\game-old.dif -ip roms\game.chd -o diff\game.dif -op roms\game.chd -c none Specifics regarding this change: Defined a new CHD version 5. New features/behaviors of this version: - support for up to 4 codecs; each block can use 1 of the 4 - new LZMA codec, which tends to do better than zlib overall - new FLAC codec, primarily used for CDs (but can be applied anywhere) - upgraded AVHuff codec now uses FLAC for encoding audio - new Huffman codec, used to catch more nearly-uncompressable blocks - compressed CHDs now use a compressed map for significant savings - CHDs now are aware of a "unit" size; each hunk holds 1 or more units (in general units map to sectors for hard disks/CDs) - diff'ing against a parent now diffs at the unit level, greatly improving compression Rewrote and modernized chd.c. CHD versions prior to 3 are unsupported, and version 3/4 CHDs are only supported for reading. Creating a new CHD now leaves the file open. Added methods to read and write at the unit and byte level, removing the need to handle this manually. Added metadata access methods that pass astrings and dynamic_buffers to simplify the interfaces. A companion class chd_compressor now implements full multithreaded compression, analyzing and compressing multiple hunks independently in parallel. Split the codec implementations out into a separate file chdcodec.* Updated harddisk.c and cdrom.c to rely on the caching/byte-level read/ write capabilities of the chd_file class. cdrom.c (and chdman) now also pad CDs to 4-frame boundaries instead of hunk boundaries, ensuring that the same SHA1 hashes are produced regardless of the hunk size. Rewrote chdman.exe entirely, switching from positional parameters to proper options. Use "chdman help" to get a list of commands, and "chdman help <command>" to get help for any particular command. Many redundant commands were removed now that additional flexibility is available. Some basic mappings: Old: chdman -createblankhd <out.chd> <cyls> <heads> <secs> New: chdman createhd -o <out.chd> -chs <cyls>,<heads>,<secs> Old: chdman -createuncomphd <in.raw> <out.chd> .... New: chdman createhd -i <in.raw> -o <out.chd> -c none .... Old: chdman -verifyfix <in.chd> New: chdman verify -i <in.chd> -f Old: chdman -merge <parent.chd> <diff.chd> <out.chd> New: chdman copy -i <diff.chd> -ip <parent.chd> -o <out.chd> Old: chdman -diff <parent.chd> <compare.chd> <diff.chd> New: chdman copy -i <compare.chd> -o <diff.chd> -op <parent.chd> Old: chdman -update <in.chd> <out.chd> New: chdman copy -i <in.chd> -o <out.chd> Added new core file coretmpl.h to hold core template classes. For now just one class, dynamic_array<> is defined, which acts like an array of a given object but which can be appended to and/or resized. Also defines dynamic_buffer as dynamic_array<UINT8> for holding an arbitrary buffer of bytes. Expect to see these used a lot. Added new core helper hashing.c/.h which defines classes for each of the common hashing methods and creator classes to wrap the computation of these hashes. A future work item is to reimplement the core emulator hashing code using these. Split bit buffer helpers out into C++ classes and into their own public header in bitstream.h. Updated huffman.c/.h to C++, and changed the interface to make it more flexible to use in nonstandard ways. Also added huffman compression of the static tree for slightly better compression rates. Created flac.c/.h as simplified C++ wrappers around the FLAC interface. A future work item is to convert the samples sound device to a modern device and leverage this for reading FLAC files. Renamed avcomp.* to avhuff.*, updated to C++, and added support for FLAC as the audio encoding mechanism. The old huffman audio is still supported for decode only. Added a variant of core_fload that loads to a dynamic_buffer. Tweaked winwork.c a bit to not limit the maximum number of processors unless the work queue was created with the WORK_QUEUE_FLAG_HIGH_FREQ option. Further adjustments here are likely going to be necessary. Fixed bug in aviio.c which caused errors when reading some AVI files.
167 lines
3.9 KiB
C
167 lines
3.9 KiB
C
/*********************************************************************
|
|
|
|
Code to interface the image code with CHD-CD core.
|
|
|
|
Based on harddriv.c by Raphael Nabet 2003
|
|
|
|
*********************************************************************/
|
|
|
|
#include "emu.h"
|
|
#include "cdrom.h"
|
|
#include "chd_cd.h"
|
|
|
|
|
|
static const char *const error_strings[] =
|
|
{
|
|
"no error",
|
|
"no drive interface",
|
|
"out of memory",
|
|
"invalid file",
|
|
"invalid parameter",
|
|
"invalid data",
|
|
"file not found",
|
|
"requires parent",
|
|
"file not writeable",
|
|
"read error",
|
|
"write error",
|
|
"codec error",
|
|
"invalid parent",
|
|
"hunk out of range",
|
|
"decompression error",
|
|
"compression error",
|
|
"can't create file",
|
|
"can't verify file"
|
|
"operation not supported",
|
|
"can't find metadata",
|
|
"invalid metadata size",
|
|
"unsupported CHD version"
|
|
};
|
|
|
|
static const char *chd_get_error_string(int chderr)
|
|
{
|
|
if ((chderr < 0 ) || (chderr >= ARRAY_LENGTH(error_strings)))
|
|
return NULL;
|
|
return error_strings[chderr];
|
|
}
|
|
|
|
|
|
static OPTION_GUIDE_START(cd_option_guide)
|
|
OPTION_INT('K', "hunksize", "Hunk Bytes")
|
|
OPTION_GUIDE_END
|
|
|
|
static const char cd_option_spec[] = "K512/1024/2048/[4096]";
|
|
|
|
// device type definition
|
|
const device_type CDROM = &device_creator<cdrom_image_device>;
|
|
|
|
//-------------------------------------------------
|
|
// cdrom_image_device - constructor
|
|
//-------------------------------------------------
|
|
|
|
cdrom_image_device::cdrom_image_device(const machine_config &mconfig, const char *tag, device_t *owner, UINT32 clock)
|
|
: device_t(mconfig, CDROM, "Cdrom", tag, owner, clock),
|
|
device_image_interface(mconfig, *this)
|
|
{
|
|
|
|
}
|
|
|
|
//-------------------------------------------------
|
|
// cdrom_image_device - destructor
|
|
//-------------------------------------------------
|
|
|
|
cdrom_image_device::~cdrom_image_device()
|
|
{
|
|
}
|
|
|
|
//-------------------------------------------------
|
|
// device_config_complete - perform any
|
|
// operations now that the configuration is
|
|
// complete
|
|
//-------------------------------------------------
|
|
|
|
void cdrom_image_device::device_config_complete()
|
|
{
|
|
// inherit a copy of the static data
|
|
const cdrom_interface *intf = reinterpret_cast<const cdrom_interface *>(static_config());
|
|
if (intf != NULL)
|
|
*static_cast<cdrom_interface *>(this) = *intf;
|
|
|
|
// or initialize to defaults if none provided
|
|
else
|
|
{
|
|
memset(&m_interface, 0, sizeof(m_interface));
|
|
memset(&m_device_displayinfo, 0, sizeof(m_device_displayinfo));
|
|
}
|
|
|
|
image_device_format *format = global_alloc_clear(image_device_format);;
|
|
format->m_index = 0;
|
|
format->m_name = "chdcd";
|
|
format->m_description = "CD-ROM drive";
|
|
format->m_extensions = "chd,cue,toc,nrg,gdi";
|
|
format->m_optspec = cd_option_spec;
|
|
format->m_next = NULL;
|
|
|
|
m_formatlist = format;
|
|
|
|
// set brief and instance name
|
|
update_names();
|
|
}
|
|
|
|
const option_guide *cdrom_image_device::create_option_guide() const
|
|
{
|
|
return cd_option_guide;
|
|
}
|
|
|
|
//-------------------------------------------------
|
|
// device_start - device-specific startup
|
|
//-------------------------------------------------
|
|
|
|
void cdrom_image_device::device_start()
|
|
{
|
|
m_cdrom_handle = NULL;
|
|
}
|
|
|
|
bool cdrom_image_device::call_load()
|
|
{
|
|
chd_error err = (chd_error)0;
|
|
chd_file *chd = NULL;
|
|
astring tempstring;
|
|
|
|
if (software_entry() == NULL)
|
|
{
|
|
if (strstr(m_image_name,".chd") && is_loaded()) {
|
|
err = m_self_chd.open( *image_core_file() ); /* CDs are never writeable */
|
|
if ( err )
|
|
goto error;
|
|
chd = &m_self_chd;
|
|
}
|
|
} else {
|
|
chd = get_disk_handle(device().machine(), device().subtag(tempstring,"cdrom"));
|
|
}
|
|
|
|
/* open the CHD file */
|
|
if (chd) {
|
|
m_cdrom_handle = cdrom_open( chd );
|
|
} else {
|
|
m_cdrom_handle = cdrom_open( m_image_name );
|
|
}
|
|
if ( ! m_cdrom_handle )
|
|
goto error;
|
|
|
|
return IMAGE_INIT_PASS;
|
|
|
|
error:
|
|
if ( chd && chd == &m_self_chd )
|
|
m_self_chd.close( );
|
|
if ( err )
|
|
seterror( IMAGE_ERROR_UNSPECIFIED, chd_get_error_string( err ) );
|
|
return IMAGE_INIT_FAIL;
|
|
}
|
|
|
|
void cdrom_image_device::call_unload()
|
|
{
|
|
assert(m_cdrom_handle);
|
|
cdrom_close(m_cdrom_handle);
|
|
m_cdrom_handle = NULL;
|
|
}
|