2021-12-25 22:06:47 +00:00
|
|
|
//
|
|
|
|
// IPF.hpp
|
|
|
|
// Clock Signal
|
|
|
|
//
|
|
|
|
// Created by Thomas Harte on 25/12/2021.
|
|
|
|
// Copyright © 2021 Thomas Harte. All rights reserved.
|
|
|
|
//
|
|
|
|
|
|
|
|
#ifndef IPF_hpp
|
|
|
|
#define IPF_hpp
|
|
|
|
|
|
|
|
#include "../DiskImage.hpp"
|
2022-01-06 22:24:31 +00:00
|
|
|
#include "../../Track/PCMTrack.hpp"
|
2021-12-25 22:06:47 +00:00
|
|
|
#include "../../../FileHolder.hpp"
|
|
|
|
#include "../../../TargetPlatforms.hpp"
|
|
|
|
|
|
|
|
#include <string>
|
2021-12-26 19:49:51 +00:00
|
|
|
#include <map>
|
2021-12-25 22:06:47 +00:00
|
|
|
|
|
|
|
namespace Storage {
|
|
|
|
namespace Disk {
|
|
|
|
|
|
|
|
/*!
|
2021-12-28 00:15:46 +00:00
|
|
|
Provides a @c DiskImage containing an IPF, which is a mixed stream of raw flux windows and
|
|
|
|
unencoded MFM sections along with gap records that can be used to record write splices, all
|
|
|
|
of which is variably clocked (albeit not at flux transition resolution; as a result IPF files tend to be
|
|
|
|
close in size to more primitive formats).
|
2021-12-25 22:06:47 +00:00
|
|
|
*/
|
|
|
|
class IPF: public DiskImage, public TargetPlatform::TypeDistinguisher {
|
|
|
|
public:
|
|
|
|
/*!
|
|
|
|
Construct an @c IPF containing content from the file with name @c file_name.
|
|
|
|
|
|
|
|
@throws Storage::FileHolder::Error::CantOpen if this file can't be opened.
|
|
|
|
@throws Error::InvalidFormat if the file doesn't appear to contain an .HFE format image.
|
|
|
|
@throws Error::UnknownVersion if the file looks correct but is an unsupported version.
|
|
|
|
*/
|
|
|
|
IPF(const std::string &file_name);
|
|
|
|
|
|
|
|
// implemented to satisfy @c Disk
|
|
|
|
HeadPosition get_maximum_head_position() final;
|
|
|
|
int get_head_count() final;
|
|
|
|
std::shared_ptr<Track> get_track_at_position(Track::Address address) final;
|
|
|
|
|
|
|
|
private:
|
|
|
|
Storage::FileHolder file_;
|
|
|
|
uint16_t seek_track(Track::Address address);
|
|
|
|
|
2021-12-26 19:49:51 +00:00
|
|
|
struct TrackDescription {
|
|
|
|
long file_offset = 0;
|
|
|
|
enum class Density {
|
|
|
|
Unknown,
|
|
|
|
Noise,
|
|
|
|
Auto,
|
|
|
|
CopylockAmiga,
|
|
|
|
CopylockAmigaNew,
|
|
|
|
CopylockST,
|
|
|
|
SpeedlockAmiga,
|
|
|
|
OldSpeedlockAmiga,
|
|
|
|
AdamBrierleyAmiga,
|
|
|
|
AdamBrierleyDensityKeyAmiga,
|
|
|
|
|
|
|
|
Max = AdamBrierleyDensityKeyAmiga
|
|
|
|
} density = Density::Unknown;
|
|
|
|
uint32_t start_bit_pos = 0;
|
|
|
|
uint32_t data_bits = 0;
|
|
|
|
uint32_t gap_bits = 0;
|
|
|
|
uint32_t block_count;
|
|
|
|
bool has_fuzzy_bits = false;
|
|
|
|
};
|
|
|
|
|
2021-12-25 22:06:47 +00:00
|
|
|
int head_count_;
|
|
|
|
int track_count_;
|
2021-12-26 19:49:51 +00:00
|
|
|
std::map<Track::Address, TrackDescription> tracks_;
|
2021-12-27 23:12:44 +00:00
|
|
|
bool is_sps_format_ = false;
|
2021-12-25 22:06:47 +00:00
|
|
|
|
|
|
|
TargetPlatform::Type target_platform_type() final {
|
2021-12-25 23:17:13 +00:00
|
|
|
return TargetPlatform::Type(platform_type_);
|
2021-12-25 22:06:47 +00:00
|
|
|
}
|
2021-12-25 23:17:13 +00:00
|
|
|
TargetPlatform::IntType platform_type_ = TargetPlatform::Amiga;
|
2022-01-06 22:24:31 +00:00
|
|
|
|
|
|
|
Time bit_length(TrackDescription::Density, int block);
|
|
|
|
void add_gap(std::vector<Storage::Disk::PCMSegment> &, Time bit_length, size_t num_bits, uint32_t value);
|
|
|
|
void add_unencoded_data(std::vector<Storage::Disk::PCMSegment> &, Time bit_length, size_t num_bits);
|
|
|
|
void add_raw_data(std::vector<Storage::Disk::PCMSegment> &, Time bit_length, size_t num_bits);
|
2021-12-25 22:06:47 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#endif /* IPF_hpp */
|