mirror of
https://github.com/holub/mame
synced 2025-05-28 00:31:33 +03:00

[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)
431 lines
12 KiB
C
431 lines
12 KiB
C
// license:BSD-3-Clause
|
|
// copyright-holders:Olivier Galibert
|
|
/*********************************************************************
|
|
|
|
formats/wd177x_dsk.h
|
|
|
|
helper for simple wd177x-formatted disk images
|
|
|
|
*********************************************************************/
|
|
|
|
#include "emu.h"
|
|
#include "formats/wd177x_dsk.h"
|
|
|
|
wd177x_format::wd177x_format(const format *_formats)
|
|
{
|
|
formats = _formats;
|
|
}
|
|
|
|
/*
|
|
Default implementation for find_size. May be overwritten by subclasses.
|
|
*/
|
|
int wd177x_format::find_size(io_generic *io, UINT32 form_factor)
|
|
{
|
|
UINT64 size = io_generic_size(io);
|
|
for(int i=0; formats[i].form_factor; i++) {
|
|
const format &f = formats[i];
|
|
if(form_factor != floppy_image::FF_UNKNOWN && form_factor != f.form_factor)
|
|
continue;
|
|
|
|
if(size == (UINT64)compute_track_size(f) * f.track_count * f.head_count)
|
|
return i;
|
|
}
|
|
return -1;
|
|
}
|
|
|
|
int wd177x_format::identify(io_generic *io, UINT32 form_factor)
|
|
{
|
|
int type = find_size(io, form_factor);
|
|
|
|
if(type != -1)
|
|
return 50;
|
|
return 0;
|
|
}
|
|
|
|
int wd177x_format::compute_track_size(const format &f) const
|
|
{
|
|
int track_size;
|
|
if(f.sector_base_size)
|
|
track_size = f.sector_base_size * f.sector_count;
|
|
else {
|
|
track_size = 0;
|
|
for(int i=0; i != f.sector_count; i++)
|
|
track_size += f.per_sector_size[i];
|
|
}
|
|
return track_size;
|
|
}
|
|
|
|
void wd177x_format::build_sector_description(const format &f, UINT8 *sectdata, desc_s *sectors) const
|
|
{
|
|
if(f.sector_base_id == -1) {
|
|
for(int i=0; i<f.sector_count; i++) {
|
|
int cur_offset = 0;
|
|
for(int j=0; j<f.sector_count; j++)
|
|
if(f.per_sector_id[j] < f.per_sector_id[i])
|
|
cur_offset += f.sector_base_size ? f.sector_base_size : f.per_sector_size[j];
|
|
sectors[i].data = sectdata + cur_offset;
|
|
sectors[i].size = f.sector_base_size ? f.sector_base_size : f.per_sector_size[i];
|
|
sectors[i].sector_id = f.per_sector_id[i];
|
|
}
|
|
} else {
|
|
int cur_offset = 0;
|
|
for(int i=0; i<f.sector_count; i++) {
|
|
sectors[i].data = sectdata + cur_offset;
|
|
sectors[i].size = f.sector_base_size ? f.sector_base_size : f.per_sector_size[i];
|
|
cur_offset += sectors[i].size;
|
|
sectors[i].sector_id = i + f.sector_base_id;
|
|
}
|
|
}
|
|
}
|
|
|
|
floppy_image_format_t::desc_e* wd177x_format::get_desc_fm(const format &f, int ¤t_size, int &end_gap_index)
|
|
{
|
|
static floppy_image_format_t::desc_e desc[23] = {
|
|
/* 00 */ { FM, 0xff, f.gap_1 },
|
|
/* 01 */ { SECTOR_LOOP_START, 0, f.sector_count-1 },
|
|
/* 02 */ { FM, 0x00, 6 },
|
|
/* 03 */ { CRC_CCITT_FM_START, 1 },
|
|
/* 04 */ { RAW, 0xf57e, 1 },
|
|
/* 05 */ { TRACK_ID_FM },
|
|
/* 06 */ { HEAD_ID_FM },
|
|
/* 07 */ { SECTOR_ID_FM },
|
|
/* 08 */ { SIZE_ID_FM },
|
|
/* 09 */ { CRC_END, 1 },
|
|
/* 10 */ { CRC, 1 },
|
|
/* 11 */ { FM, 0xff, f.gap_2 },
|
|
/* 12 */ { FM, 0x00, 6 },
|
|
/* 13 */ { CRC_CCITT_FM_START, 2 },
|
|
/* 14 */ { RAW, 0xf56f, 1 },
|
|
/* 15 */ { SECTOR_DATA_FM, -1 },
|
|
/* 16 */ { CRC_END, 2 },
|
|
/* 17 */ { CRC, 2 },
|
|
/* 18 */ { FM, 0xff, f.gap_3 },
|
|
/* 19 */ { SECTOR_LOOP_END },
|
|
/* 20 */ { FM, 0xff, 0 },
|
|
/* 21 */ { RAWBITS, 0xffff, 0 },
|
|
/* 22 */ { END }
|
|
};
|
|
|
|
current_size = f.gap_1*16;
|
|
if(f.sector_base_size)
|
|
current_size += f.sector_base_size * f.sector_count * 16;
|
|
else {
|
|
for(int j=0; j != f.sector_count; j++)
|
|
current_size += f.per_sector_size[j] * 16;
|
|
}
|
|
current_size += (6+1+4+2+f.gap_2+6+1+2+f.gap_3) * f.sector_count * 16;
|
|
|
|
end_gap_index = 20;
|
|
|
|
return desc;
|
|
}
|
|
|
|
floppy_image_format_t::desc_e* wd177x_format::get_desc_mfm(const format &f, int ¤t_size, int &end_gap_index)
|
|
{
|
|
static floppy_image_format_t::desc_e desc[25] = {
|
|
/* 00 */ { MFM, 0x4e, f.gap_1 },
|
|
/* 01 */ { SECTOR_LOOP_START, 0, f.sector_count-1 },
|
|
/* 02 */ { MFM, 0x00, 12 },
|
|
/* 03 */ { CRC_CCITT_START, 1 },
|
|
/* 04 */ { RAW, 0x4489, 3 },
|
|
/* 05 */ { MFM, 0xfe, 1 },
|
|
/* 06 */ { TRACK_ID },
|
|
/* 07 */ { HEAD_ID },
|
|
/* 08 */ { SECTOR_ID },
|
|
/* 09 */ { SIZE_ID },
|
|
/* 10 */ { CRC_END, 1 },
|
|
/* 11 */ { CRC, 1 },
|
|
/* 12 */ { MFM, 0x4e, f.gap_2 },
|
|
/* 13 */ { MFM, 0x00, 12 },
|
|
/* 14 */ { CRC_CCITT_START, 2 },
|
|
/* 15 */ { RAW, 0x4489, 3 },
|
|
/* 16 */ { MFM, 0xfb, 1 },
|
|
/* 17 */ { SECTOR_DATA, -1 },
|
|
/* 18 */ { CRC_END, 2 },
|
|
/* 19 */ { CRC, 2 },
|
|
/* 20 */ { MFM, 0x4e, f.gap_3 },
|
|
/* 21 */ { SECTOR_LOOP_END },
|
|
/* 22 */ { MFM, 0x4e, 0 },
|
|
/* 23 */ { RAWBITS, 0x9254, 0 },
|
|
/* 24 */ { END }
|
|
};
|
|
|
|
current_size = f.gap_1*16;
|
|
if(f.sector_base_size)
|
|
current_size += f.sector_base_size * f.sector_count * 16;
|
|
else {
|
|
for(int j=0; j != f.sector_count; j++)
|
|
current_size += f.per_sector_size[j] * 16;
|
|
}
|
|
current_size += (12+3+1+4+2+f.gap_2+12+3+1+2+f.gap_3) * f.sector_count * 16;
|
|
|
|
end_gap_index = 22;
|
|
|
|
return desc;
|
|
}
|
|
|
|
bool wd177x_format::load(io_generic *io, UINT32 form_factor, floppy_image *image)
|
|
{
|
|
int type = find_size(io, form_factor);
|
|
if(type == -1)
|
|
return false;
|
|
|
|
const format &f = formats[type];
|
|
floppy_image_format_t::desc_e *desc;
|
|
int current_size;
|
|
int end_gap_index;
|
|
|
|
switch (f.encoding)
|
|
{
|
|
case floppy_image::FM:
|
|
desc = get_desc_fm(f, current_size, end_gap_index);
|
|
break;
|
|
case floppy_image::MFM:
|
|
default:
|
|
desc = get_desc_mfm(f, current_size, end_gap_index);
|
|
break;
|
|
}
|
|
|
|
int total_size = 200000000/f.cell_size;
|
|
int remaining_size = total_size - current_size;
|
|
if(remaining_size < 0)
|
|
throw emu_fatalerror("wd177x_format: Incorrect track layout, max_size=%d, current_size=%d", total_size, current_size);
|
|
|
|
// Fixup the end gap
|
|
desc[end_gap_index].p2 = remaining_size / 16;
|
|
desc[end_gap_index + 1].p2 = remaining_size & 15;
|
|
desc[end_gap_index + 1].p1 >>= 16-(remaining_size & 15);
|
|
|
|
int track_size = compute_track_size(f);
|
|
|
|
UINT8 sectdata[40*512];
|
|
desc_s sectors[40];
|
|
build_sector_description(f, sectdata, sectors);
|
|
|
|
for(int track=0; track < f.track_count; track++)
|
|
for(int head=0; head < f.head_count; head++) {
|
|
io_generic_read(io, sectdata, get_image_offset(f, head, track), track_size);
|
|
generate_track(desc, track, head, sectors, f.sector_count, total_size, image);
|
|
}
|
|
|
|
image->set_variant(f.variant);
|
|
|
|
return true;
|
|
}
|
|
|
|
bool wd177x_format::supports_save() const
|
|
{
|
|
return true;
|
|
}
|
|
|
|
bool wd177x_format::save(io_generic *io, floppy_image *image)
|
|
{
|
|
// Count the number of formats
|
|
int formats_count;
|
|
for(formats_count=0; formats[formats_count].form_factor; formats_count++);
|
|
|
|
// Allocate the storage for the list of testable formats for a
|
|
// given cell size
|
|
dynamic_array<int> candidates(formats_count);
|
|
|
|
// Format we're finally choosing
|
|
int chosen_candidate = -1;
|
|
|
|
// Previously tested cell size
|
|
int min_cell_size = 0;
|
|
for(;;) {
|
|
// Build the list of all formats for the immediatly superior cell size
|
|
int cur_cell_size = 0;
|
|
int candidates_count = 0;
|
|
for(int i=0; i != formats_count; i++) {
|
|
if(image->get_form_factor() == floppy_image::FF_UNKNOWN ||
|
|
image->get_form_factor() == formats[i].form_factor) {
|
|
if(formats[i].cell_size == cur_cell_size)
|
|
candidates[candidates_count++] = i;
|
|
else if((!cur_cell_size || formats[i].cell_size < cur_cell_size) &&
|
|
formats[i].cell_size > min_cell_size) {
|
|
candidates[0] = i;
|
|
candidates_count = 1;
|
|
cur_cell_size = formats[i].cell_size;
|
|
}
|
|
}
|
|
}
|
|
|
|
min_cell_size = cur_cell_size;
|
|
|
|
// No candidates with a cell size bigger than the previously
|
|
// tested one, we're done
|
|
if(!candidates_count)
|
|
break;
|
|
|
|
// Filter with track 0 head 0
|
|
check_compatibility(image, candidates, candidates_count);
|
|
|
|
// Nobody matches, try with the next cell size
|
|
if(!candidates_count)
|
|
continue;
|
|
|
|
// We have a match at that cell size, we just need to find the
|
|
// best one given the geometry
|
|
|
|
// If there's only one, we're done
|
|
if(candidates_count == 1) {
|
|
chosen_candidate = candidates[0];
|
|
break;
|
|
}
|
|
|
|
// Otherwise, find the best
|
|
int tracks, heads;
|
|
image->get_actual_geometry(tracks, heads);
|
|
chosen_candidate = candidates[0];
|
|
for(int i=1; i != candidates_count; i++) {
|
|
const format &cc = formats[chosen_candidate];
|
|
const format &cn = formats[candidates[i]];
|
|
|
|
// Handling enough sides is better than not
|
|
if(cn.head_count >= heads && cc.head_count < heads)
|
|
goto change;
|
|
else if(cc.head_count >= heads && cn.head_count < heads)
|
|
goto dont_change;
|
|
|
|
// Since we're limited to two heads, at that point head
|
|
// count is identical for both formats.
|
|
|
|
// Handling enough tracks is better than not
|
|
if(cn.track_count >= tracks && cc.track_count < tracks)
|
|
goto change;
|
|
else if(cn.track_count >= tracks && cc.track_count < tracks)
|
|
goto dont_change;
|
|
|
|
// Both are on the same side of the track count, so closest is best
|
|
if(cc.track_count < tracks && cn.track_count > cc.track_count)
|
|
goto change;
|
|
if(cc.track_count >= tracks && cn.track_count < cc.track_count)
|
|
goto change;
|
|
goto dont_change;
|
|
|
|
change:
|
|
chosen_candidate = candidates[i];
|
|
dont_change:
|
|
;
|
|
}
|
|
// We have a winner, bail out
|
|
break;
|
|
}
|
|
|
|
// No match, pick the first one and be done with it
|
|
if(chosen_candidate == -1)
|
|
chosen_candidate = 0;
|
|
|
|
|
|
const format &f = formats[chosen_candidate];
|
|
int track_size = compute_track_size(f);
|
|
|
|
UINT8 sectdata[40*512];
|
|
desc_s sectors[40];
|
|
build_sector_description(f, sectdata, sectors);
|
|
|
|
for(int track=0; track < f.track_count; track++)
|
|
for(int head=0; head < f.head_count; head++) {
|
|
extract_sectors(image, f, sectors, track, head);
|
|
io_generic_write(io, sectdata, get_image_offset(f, head, track), track_size);
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
/*
|
|
Default implementation of the image offset computation. May be overwritten
|
|
by subclasses.
|
|
*/
|
|
int wd177x_format::get_image_offset(const format &f, int head, int track)
|
|
{
|
|
return (track * f.head_count + head) * compute_track_size(f);
|
|
}
|
|
|
|
void wd177x_format::check_compatibility(floppy_image *image, int *candidates, int &candidates_count)
|
|
{
|
|
UINT8 bitstream[500000/8];
|
|
UINT8 sectdata[50000];
|
|
desc_xs sectors[256];
|
|
int track_size;
|
|
|
|
// Extract the sectors
|
|
generate_bitstream_from_track(0, 0, formats[candidates[0]].cell_size, bitstream, track_size, image);
|
|
|
|
switch (formats[candidates[0]].encoding)
|
|
{
|
|
case floppy_image::FM:
|
|
extract_sectors_from_bitstream_fm_pc(bitstream, track_size, sectors, sectdata, sizeof(sectdata));
|
|
break;
|
|
case floppy_image::MFM:
|
|
extract_sectors_from_bitstream_mfm_pc(bitstream, track_size, sectors, sectdata, sizeof(sectdata));
|
|
break;
|
|
}
|
|
|
|
// Check compatibility with every candidate, copy in-place
|
|
int *ok_cands = candidates;
|
|
for(int i=0; i != candidates_count; i++) {
|
|
const format &f = formats[candidates[i]];
|
|
int ns = 0;
|
|
for(int j=0; j<256; j++)
|
|
if(sectors[j].data) {
|
|
int sid;
|
|
if(f.sector_base_id == -1) {
|
|
for(sid=0; sid < f.sector_count; sid++)
|
|
if(f.per_sector_id[sid] == j)
|
|
break;
|
|
} else
|
|
sid = j - f.sector_base_id;
|
|
if(sid < 0 || sid > f.sector_count)
|
|
goto fail;
|
|
if(f.sector_base_size) {
|
|
if(sectors[j].size != f.sector_base_size)
|
|
goto fail;
|
|
} else {
|
|
if(sectors[j].size != f.per_sector_size[sid])
|
|
goto fail;
|
|
}
|
|
ns++;
|
|
}
|
|
if(ns == f.sector_count)
|
|
*ok_cands++ = candidates[i];
|
|
fail:
|
|
;
|
|
}
|
|
candidates_count = ok_cands - candidates;
|
|
}
|
|
|
|
void wd177x_format::extract_sectors(floppy_image *image, const format &f, desc_s *sdesc, int track, int head)
|
|
{
|
|
UINT8 bitstream[500000/8];
|
|
UINT8 sectdata[50000];
|
|
desc_xs sectors[256];
|
|
int track_size;
|
|
|
|
// Extract the sectors
|
|
generate_bitstream_from_track(track, head, f.cell_size, bitstream, track_size, image);
|
|
|
|
switch (f.encoding)
|
|
{
|
|
case floppy_image::FM:
|
|
extract_sectors_from_bitstream_fm_pc(bitstream, track_size, sectors, sectdata, sizeof(sectdata));
|
|
break;
|
|
case floppy_image::MFM:
|
|
extract_sectors_from_bitstream_mfm_pc(bitstream, track_size, sectors, sectdata, sizeof(sectdata));
|
|
break;
|
|
}
|
|
|
|
for(int i=0; i<f.sector_count; i++) {
|
|
desc_s &ds = sdesc[i];
|
|
desc_xs &xs = sectors[ds.sector_id];
|
|
if(!xs.data)
|
|
memset((void *)ds.data, 0, ds.size);
|
|
else if(xs.size < ds.size) {
|
|
memcpy((void *)ds.data, xs.data, xs.size);
|
|
memset((UINT8 *)ds.data + xs.size, 0, xs.size - ds.size);
|
|
} else
|
|
memcpy((void *)ds.data, xs.data, ds.size);
|
|
}
|
|
}
|