Files
fluxengine/lib/imagereader/nfdimagereader.cc

171 lines
6.2 KiB
C++

#include "lib/core/globals.h"
#include "lib/config/flags.h"
#include "lib/data/sector.h"
#include "lib/imagereader/imagereader.h"
#include "lib/data/image.h"
#include "lib/config/proto.h"
#include "lib/core/logger.h"
#include "lib/config/config.pb.h"
#include <algorithm>
#include <iostream>
#include <fstream>
// reader based on this partial documentation of the D88 format:
// https://www.pc98.org/project/doc/d88.html
class NFDImageReader : public ImageReader
{
public:
NFDImageReader(const ImageReaderProto& config): ImageReader(config) {}
std::unique_ptr<Image> readImage() override
{
std::ifstream inputFile(
_config.filename(), std::ios::in | std::ios::binary);
if (!inputFile.is_open())
error("cannot open input file");
Bytes fileId(14); // read first entry of track table as well
inputFile.read((char*)fileId.begin(), fileId.size());
if (fileId == Bytes("T98FDDIMAGE.R1"))
{
error("NFD: r1 images are not currently supported");
}
if (fileId != Bytes("T98FDDIMAGE.R0"))
{
error("NFD: could not find NFD header");
}
Bytes header(0x10a10);
inputFile.seekg(0, std::ios::beg);
inputFile.read((char*)header.begin(), header.size());
ByteReader headerReader(header);
char heads = headerReader.seek(0x115).read_8();
if (heads != 2)
{
error("NFD: unsupported number of heads");
}
if (_extraConfig.encoder().format_case() !=
EncoderProto::FormatCase::FORMAT_NOT_SET)
log("NFD: overriding configured format");
auto ibm = _extraConfig.mutable_encoder()->mutable_ibm();
auto layout = _extraConfig.mutable_layout();
log("NFD: HD 1.2MB mode");
log("NFD: forcing hign density mode");
_extraConfig.mutable_drive()->set_high_density(true);
_extraConfig.mutable_layout()->set_format_type(FORMATTYPE_80TRACK);
std::unique_ptr<Image> image(new Image);
for (int track = 0; track < 163; track++)
{
auto trackdata = ibm->add_trackdata();
trackdata->set_target_clock_period_us(2);
trackdata->set_target_rotational_period_ms(167);
auto layoutdata = layout->add_layoutdata();
auto physical = layoutdata->mutable_physical();
int currentTrackTrack = -1;
int currentTrackHead = -1;
int trackSectorSize = -1;
for (int sectorInTrack = 0; sectorInTrack < 26; sectorInTrack++)
{
headerReader.seek(0x120 + track * 26 * 16 + sectorInTrack * 16);
int track = headerReader.read_8();
int head = headerReader.read_8();
int sectorId = headerReader.read_8();
int sectorSize = 128 << headerReader.read_8();
int mfm = headerReader.read_8();
int ddam = headerReader.read_8();
int status = headerReader.read_8();
headerReader.skip(9); // skip ST0, ST1, ST2, PDA, reserved(5)
if (track == 0xFF)
continue;
if (ddam != 0)
error("NFD: nonzero ddam currently unsupported");
if (status != 0)
error(
"NFD: nonzero fdd status codes are currently "
"unsupported");
if (currentTrackTrack < 0)
{
currentTrackTrack = track;
currentTrackHead = head;
}
else if (currentTrackTrack != track)
{
error(
"NFD: all sectors in a track must belong to the same "
"track");
}
else if (currentTrackHead != head)
{
error(
"NFD: all sectors in a track must belong to the same "
"head");
}
if (trackSectorSize < 0)
{
trackSectorSize = sectorSize;
// this is the first sector we've read, use it settings for
// per-track data
trackdata->set_track(track);
trackdata->set_head(head);
layoutdata->set_track(track);
layoutdata->set_side(head);
layoutdata->set_sector_size(sectorSize);
trackdata->set_use_fm(!mfm);
if (!mfm)
{
trackdata->set_gap_fill_byte(0xffff);
trackdata->set_idam_byte(0xf57e);
trackdata->set_dam_byte(0xf56f);
}
// create timings to approximately match N88-BASIC
if (sectorSize <= 128)
{
trackdata->set_gap0(0x1b);
trackdata->set_gap2(0x09);
trackdata->set_gap3(0x1b);
}
else if (sectorSize <= 256)
{
trackdata->set_gap0(0x36);
trackdata->set_gap3(0x36);
}
}
else if (trackSectorSize != sectorSize)
{
error(
"NFD: multiple sector sizes per track are currently "
"unsupported");
}
Bytes data(sectorSize);
inputFile.read((char*)data.begin(), data.size());
physical->add_sector(sectorId);
const auto& sector = image->put(track, head, sectorId);
sector->status = Sector::OK;
sector->data = data;
}
}
image->calculateSize();
const Geometry& geometry = image->getGeometry();
log("NFD: read {} tracks, {} sides",
geometry.numTracks,
geometry.numSides);
return image;
}
};
std::unique_ptr<ImageReader> ImageReader::createNFDImageReader(
const ImageReaderProto& config)
{
return std::unique_ptr<ImageReader>(new NFDImageReader(config));
}