mirror of
				https://github.com/davidgiven/fluxengine.git
				synced 2025-10-31 11:17:01 -07:00 
			
		
		
		
	
		
			
				
	
	
		
			138 lines
		
	
	
		
			5.4 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			138 lines
		
	
	
		
			5.4 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
#include "globals.h"
 | 
						|
#include "flags.h"
 | 
						|
#include "sector.h"
 | 
						|
#include "imagewriter/imagewriter.h"
 | 
						|
#include "ldbs.h"
 | 
						|
#include "image.h"
 | 
						|
#include "logger.h"
 | 
						|
#include "lib/config.pb.h"
 | 
						|
#include <algorithm>
 | 
						|
#include <iostream>
 | 
						|
#include <fstream>
 | 
						|
 | 
						|
class LDBSImageWriter : public ImageWriter
 | 
						|
{
 | 
						|
public:
 | 
						|
    LDBSImageWriter(const ImageWriterProto& config): ImageWriter(config) {}
 | 
						|
 | 
						|
    void writeImage(const Image& image)
 | 
						|
    {
 | 
						|
        LDBS ldbs;
 | 
						|
 | 
						|
        const Geometry geometry = image.getGeometry();
 | 
						|
 | 
						|
        log("LDBS: writing {} tracks, {} sides, {} sectors, {} bytes per "
 | 
						|
            "sector",
 | 
						|
            geometry.numTracks,
 | 
						|
            geometry.numSides,
 | 
						|
            geometry.numSectors,
 | 
						|
            geometry.sectorSize);
 | 
						|
 | 
						|
        Bytes trackDirectory;
 | 
						|
        ByteWriter trackDirectoryWriter(trackDirectory);
 | 
						|
        int trackDirectorySize = 0;
 | 
						|
        trackDirectoryWriter.write_le16(0);
 | 
						|
 | 
						|
        LDBSOutputProto::DataRate dataRate = _config.ldbs().data_rate();
 | 
						|
        if (dataRate == LDBSOutputProto::RATE_GUESS)
 | 
						|
        {
 | 
						|
            dataRate = (geometry.numSectors > 10) ? LDBSOutputProto::RATE_HD
 | 
						|
                                                  : LDBSOutputProto::RATE_DD;
 | 
						|
            if (geometry.sectorSize <= 256)
 | 
						|
                dataRate = LDBSOutputProto::RATE_SD;
 | 
						|
            log("LDBS: guessing data rate as {}",
 | 
						|
                LDBSOutputProto::DataRate_Name(dataRate));
 | 
						|
        }
 | 
						|
 | 
						|
        LDBSOutputProto::RecordingMode recordingMode =
 | 
						|
            _config.ldbs().recording_mode();
 | 
						|
        if (recordingMode == LDBSOutputProto::RECMODE_GUESS)
 | 
						|
        {
 | 
						|
            recordingMode = LDBSOutputProto::RECMODE_MFM;
 | 
						|
            log("LDBS: guessing recording mode as {}",
 | 
						|
                LDBSOutputProto::RecordingMode_Name(recordingMode));
 | 
						|
        }
 | 
						|
 | 
						|
        for (int track = 0; track < geometry.numTracks; track++)
 | 
						|
        {
 | 
						|
            for (int side = 0; side < geometry.numSides; side++)
 | 
						|
            {
 | 
						|
                Bytes trackHeader;
 | 
						|
                ByteWriter trackHeaderWriter(trackHeader);
 | 
						|
 | 
						|
                int actualSectors = 0;
 | 
						|
                for (int sectorId = 0; sectorId < geometry.numSectors;
 | 
						|
                     sectorId++)
 | 
						|
                {
 | 
						|
                    const auto& sector = image.get(track, side, sectorId);
 | 
						|
                    if (sector)
 | 
						|
                        actualSectors++;
 | 
						|
                }
 | 
						|
 | 
						|
                trackHeaderWriter.write_le16(
 | 
						|
                    0x000C); /* offset of sector sideers */
 | 
						|
                trackHeaderWriter.write_le16(
 | 
						|
                    0x0012); /* length of each sector descriptor */
 | 
						|
                trackHeaderWriter.write_le16(actualSectors);
 | 
						|
                trackHeaderWriter.write_8(dataRate);
 | 
						|
                trackHeaderWriter.write_8(recordingMode);
 | 
						|
                trackHeaderWriter.write_8(0);    /* format gap length */
 | 
						|
                trackHeaderWriter.write_8(0);    /* filler byte */
 | 
						|
                trackHeaderWriter.write_le16(0); /* approximate track length */
 | 
						|
 | 
						|
                for (int sectorId = 0; sectorId < geometry.numSectors;
 | 
						|
                     sectorId++)
 | 
						|
                {
 | 
						|
                    const auto& sector = image.get(track, side, sectorId);
 | 
						|
                    if (sector)
 | 
						|
                    {
 | 
						|
                        uint32_t sectorLabel = (('S') << 24) |
 | 
						|
                                               ((track & 0xff) << 16) |
 | 
						|
                                               (side << 8) | sectorId;
 | 
						|
                        uint32_t sectorAddress =
 | 
						|
                            ldbs.put(sector->data, sectorLabel);
 | 
						|
 | 
						|
                        trackHeaderWriter.write_8(track);
 | 
						|
                        trackHeaderWriter.write_8(side);
 | 
						|
                        trackHeaderWriter.write_8(sectorId);
 | 
						|
                        trackHeaderWriter.write_8(0); /* power-of-two size */
 | 
						|
                        trackHeaderWriter.write_8(
 | 
						|
                            (sector->status == Sector::OK)
 | 
						|
                                ? 0x00
 | 
						|
                                : 0x20);              /* 8272 status 1 */
 | 
						|
                        trackHeaderWriter.write_8(0); /* 8272 status 2 */
 | 
						|
                        trackHeaderWriter.write_8(1); /* number of copies */
 | 
						|
                        trackHeaderWriter.write_8(0); /* filler byte */
 | 
						|
                        trackHeaderWriter.write_le32(sectorAddress);
 | 
						|
                        trackHeaderWriter.write_le16(0); /* trailing bytes */
 | 
						|
                        trackHeaderWriter.write_le16(
 | 
						|
                            0); /* approximate offset */
 | 
						|
                        trackHeaderWriter.write_le16(sector->data.size());
 | 
						|
                    }
 | 
						|
                }
 | 
						|
 | 
						|
                uint32_t trackLabel = (('T') << 24) | ((track & 0xff) << 16) |
 | 
						|
                                      ((track >> 8) << 8) | side;
 | 
						|
                uint32_t trackHeaderAddress = ldbs.put(trackHeader, trackLabel);
 | 
						|
                trackDirectoryWriter.write_be32(trackLabel);
 | 
						|
                trackDirectoryWriter.write_le32(trackHeaderAddress);
 | 
						|
                trackDirectorySize++;
 | 
						|
            }
 | 
						|
        }
 | 
						|
 | 
						|
        trackDirectoryWriter.seek(0);
 | 
						|
        trackDirectoryWriter.write_le16(trackDirectorySize);
 | 
						|
 | 
						|
        uint32_t trackDirectoryAddress =
 | 
						|
            ldbs.put(trackDirectory, LDBS_TRACK_BLOCK);
 | 
						|
        Bytes data = ldbs.write(trackDirectoryAddress);
 | 
						|
        data.writeToFile(_config.filename());
 | 
						|
    }
 | 
						|
};
 | 
						|
 | 
						|
std::unique_ptr<ImageWriter> ImageWriter::createLDBSImageWriter(
 | 
						|
    const ImageWriterProto& config)
 | 
						|
{
 | 
						|
    return std::unique_ptr<ImageWriter>(new LDBSImageWriter(config));
 | 
						|
}
 |