mirror of
https://github.com/TomHarte/CLK.git
synced 2025-01-16 11:30:22 +00:00
58 lines
1.6 KiB
C++
58 lines
1.6 KiB
C++
|
//
|
||
|
// FAT12.cpp
|
||
|
// Clock Signal
|
||
|
//
|
||
|
// Created by Thomas Harte on 07/01/2018.
|
||
|
// Copyright 2018 Thomas Harte. All rights reserved.
|
||
|
//
|
||
|
|
||
|
#include "FAT12.hpp"
|
||
|
|
||
|
#include "Utility/ImplicitSectors.hpp"
|
||
|
|
||
|
using namespace Storage::Disk;
|
||
|
|
||
|
FAT12::FAT12(const std::string &file_name) :
|
||
|
MFMSectorDump(file_name) {
|
||
|
// The only sanity check here is whether a sensible
|
||
|
// geometry is encoded in the first sector, or can be guessed.
|
||
|
off_t file_size = file_.stats().st_size;
|
||
|
|
||
|
if(file_size < 512) throw Error::InvalidFormat;
|
||
|
|
||
|
// Inspect the FAT.
|
||
|
file_.seek(11, SEEK_SET);
|
||
|
sector_size_ = file_.get16le();
|
||
|
file_.seek(19, SEEK_SET);
|
||
|
const uint16_t total_sectors = file_.get16le();
|
||
|
file_.seek(24, SEEK_SET);
|
||
|
sector_count_ = file_.get16le();
|
||
|
head_count_ = file_.get16le();
|
||
|
|
||
|
// Throw if there would seemingly be an incomplete track.
|
||
|
if(file_size != total_sectors*sector_size_) throw Error::InvalidFormat;
|
||
|
if(total_sectors % (head_count_ * sector_count_)) throw Error::InvalidFormat;
|
||
|
track_count_ = int(total_sectors / (head_count_ * sector_count_));
|
||
|
|
||
|
// Check that there is a valid power-of-two sector size.
|
||
|
uint8_t log_sector_size = 2;
|
||
|
while(log_sector_size < 5 && (1 << (7+log_sector_size)) != sector_size_) {
|
||
|
++log_sector_size;
|
||
|
}
|
||
|
if(log_sector_size >= 5) throw Error::InvalidFormat;
|
||
|
|
||
|
set_geometry(sector_count_, log_sector_size, 1, true);
|
||
|
}
|
||
|
|
||
|
HeadPosition FAT12::get_maximum_head_position() {
|
||
|
return HeadPosition(track_count_);
|
||
|
}
|
||
|
|
||
|
int FAT12::get_head_count() {
|
||
|
return head_count_;
|
||
|
}
|
||
|
|
||
|
long FAT12::get_file_offset_for_position(Track::Address address) {
|
||
|
return (address.position.as_int()*head_count_ + address.head) * sector_size_ * sector_count_;
|
||
|
}
|