mirror of
				https://github.com/davidgiven/fluxengine.git
				synced 2025-10-24 11:11:02 -07:00 
			
		
		
		
	
		
			
				
	
	
		
			171 lines
		
	
	
		
			6.2 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			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));
 | |
| }
 |