mirror of
				https://github.com/davidgiven/fluxengine.git
				synced 2025-10-24 11:11:02 -07:00 
			
		
		
		
	
		
			
				
	
	
		
			131 lines
		
	
	
		
			4.4 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			131 lines
		
	
	
		
			4.4 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| #include "lib/core/globals.h"
 | |
| #include "lib/config/config.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 FDI format:
 | |
| // https://www.pc98.org/project/doc/hdi.html
 | |
| 
 | |
| class FdiImageReader : public ImageReader
 | |
| {
 | |
| public:
 | |
|     FdiImageReader(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 header(32);
 | |
|         inputFile.read((char*)header.begin(), header.size());
 | |
|         ByteReader headerReader(header);
 | |
|         if (headerReader.seek(0).read_le32() != 0)
 | |
|             error("FDI: could not find FDI header, is this a FDI file?");
 | |
| 
 | |
|         // we currently don't use fddType but it could be used to automatically
 | |
|         // select profile parameters in the future
 | |
|         //
 | |
|         int fddType = headerReader.seek(4).read_le32();
 | |
|         int headerSize = headerReader.seek(0x08).read_le32();
 | |
|         int sectorSize = headerReader.seek(0x10).read_le32();
 | |
|         int sectorsPerTrack = headerReader.seek(0x14).read_le32();
 | |
|         int sides = headerReader.seek(0x18).read_le32();
 | |
|         int tracks = headerReader.seek(0x1c).read_le32();
 | |
| 
 | |
|         inputFile.seekg(headerSize);
 | |
| 
 | |
|         std::unique_ptr<Image> image(new Image);
 | |
|         int trackCount = 0;
 | |
|         for (int track = 0; track < tracks; track++)
 | |
|         {
 | |
|             if (inputFile.eof())
 | |
|                 break;
 | |
| 
 | |
|             for (int side = 0; side < sides; side++)
 | |
|             {
 | |
|                 std::vector<unsigned> sectors;
 | |
|                 for (int i = 0; i < sectorsPerTrack; i++)
 | |
|                     sectors.push_back(i + 1);
 | |
| 
 | |
|                 for (int sectorId : sectors)
 | |
|                 {
 | |
|                     Bytes data(sectorSize);
 | |
|                     inputFile.read((char*)data.begin(), data.size());
 | |
| 
 | |
|                     const auto& sector = image->put(track, side, sectorId);
 | |
|                     sector->status = Sector::OK;
 | |
|                     sector->data = data;
 | |
|                 }
 | |
|             }
 | |
| 
 | |
|             trackCount++;
 | |
|         }
 | |
| 
 | |
|         auto layout = _extraConfig.mutable_layout();
 | |
|         if (globalConfig()->encoder().format_case() ==
 | |
|             EncoderProto::FormatCase::FORMAT_NOT_SET)
 | |
|         {
 | |
|             auto ibm = _extraConfig.mutable_encoder()->mutable_ibm();
 | |
|             auto trackdata = ibm->add_trackdata();
 | |
|             trackdata->set_target_clock_period_us(2);
 | |
| 
 | |
|             auto layoutdata = layout->add_layoutdata();
 | |
|             auto physical = layoutdata->mutable_physical();
 | |
|             switch (fddType)
 | |
|             {
 | |
|                 case 0x90:
 | |
|                     log("FDI: automatically setting format to 1.2MB (1024 byte "
 | |
|                         "sectors)");
 | |
|                     trackdata->set_target_rotational_period_ms(167);
 | |
|                     layoutdata->set_sector_size(1024);
 | |
|                     for (int i = 0; i < 9; i++)
 | |
|                         physical->add_sector(i);
 | |
|                     break;
 | |
| 
 | |
|                 case 0x30:
 | |
|                     log("FDI: automatically setting format to 1.44MB");
 | |
|                     trackdata->set_target_rotational_period_ms(200);
 | |
|                     layoutdata->set_sector_size(512);
 | |
|                     for (int i = 0; i < 18; i++)
 | |
|                         physical->add_sector(i);
 | |
|                     break;
 | |
| 
 | |
|                 default:
 | |
|                     error(
 | |
|                         "FDI: unknown fdd type 0x{:2x}, could not determine "
 | |
|                         "write profile automatically",
 | |
|                         fddType);
 | |
|                     break;
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         image->calculateSize();
 | |
|         const Geometry& geometry = image->getGeometry();
 | |
|         log("FDI: read {} tracks, {} sides, {} kB total",
 | |
|             geometry.numTracks,
 | |
|             geometry.numSides,
 | |
|             ((int)inputFile.tellg() - headerSize) / 1024);
 | |
| 
 | |
|         layout->set_tracks(geometry.numTracks);
 | |
|         layout->set_sides(geometry.numSides);
 | |
| 
 | |
|         return image;
 | |
|     }
 | |
| };
 | |
| 
 | |
| std::unique_ptr<ImageReader> ImageReader::createFdiImageReader(
 | |
|     const ImageReaderProto& config)
 | |
| {
 | |
|     return std::unique_ptr<ImageReader>(new FdiImageReader(config));
 | |
| }
 |