Duckstation/src/common/cd_image.h

264 lines
7.2 KiB
C
Raw Normal View History

2019-09-20 10:14:00 +00:00
#pragma once
#include "bitfield.h"
2019-09-20 10:14:00 +00:00
#include "types.h"
#include <memory>
#include <string>
2019-10-17 13:54:51 +00:00
#include <tuple>
#include <vector>
2019-09-20 10:14:00 +00:00
class CDImage
{
public:
CDImage();
virtual ~CDImage();
using LBA = u32;
2019-09-20 10:14:00 +00:00
2019-09-21 15:12:16 +00:00
enum : u32
{
RAW_SECTOR_SIZE = 2352,
DATA_SECTOR_SIZE = 2048,
SECTOR_SYNC_SIZE = 12,
SECTOR_HEADER_SIZE = 4,
FRAMES_PER_SECOND = 75, // "sectors", or "timecode frames" (not "channel frames")
2019-09-21 15:12:16 +00:00
SECONDS_PER_MINUTE = 60,
FRAMES_PER_MINUTE = FRAMES_PER_SECOND * SECONDS_PER_MINUTE,
SUBCHANNEL_BYTES_PER_FRAME = 12,
LEAD_OUT_SECTOR_COUNT = 6750
};
enum : u8
{
LEAD_OUT_TRACK_NUMBER = 0xAA
2019-09-21 15:12:16 +00:00
};
2019-09-20 10:14:00 +00:00
enum class ReadMode : u32
{
DataOnly, // 2048 bytes per sector.
RawSector, // 2352 bytes per sector.
RawNoSync, // 2340 bytes per sector.
};
enum class TrackMode : u32
{
Audio, // 2352 bytes per sector
Mode1, // 2048 bytes per sector
Mode1Raw, // 2352 bytes per sector
Mode2, // 2336 bytes per sector
Mode2Form1, // 2048 bytes per sector
Mode2Form2, // 2324 bytes per sector
Mode2FormMix, // 2332 bytes per sector
Mode2Raw // 2352 bytes per sector
};
struct SectorHeader
{
u8 minute;
u8 second;
u8 frame;
u8 sector_mode;
};
struct Position
2019-10-17 13:54:51 +00:00
{
u8 minute;
u8 second;
u8 frame;
static constexpr Position FromBCD(u8 minute, u8 second, u8 frame)
{
return Position{PackedBCDToBinary(minute), PackedBCDToBinary(second), PackedBCDToBinary(frame)};
}
static constexpr Position FromLBA(LBA lba)
{
const u8 frame = Truncate8(lba % FRAMES_PER_SECOND);
lba /= FRAMES_PER_SECOND;
const u8 second = Truncate8(lba % SECONDS_PER_MINUTE);
lba /= SECONDS_PER_MINUTE;
const u8 minute = Truncate8(lba);
return Position{minute, second, frame};
}
LBA ToLBA() const
{
return ZeroExtend32(minute) * FRAMES_PER_MINUTE + ZeroExtend32(second) * FRAMES_PER_SECOND + ZeroExtend32(frame);
}
constexpr std::tuple<u8, u8, u8> ToBCD() const
{
return std::make_tuple<u8, u8, u8>(BinaryToBCD(minute), BinaryToBCD(second), BinaryToBCD(frame));
}
Position operator+(const Position& rhs) { return FromLBA(ToLBA() + rhs.ToLBA()); }
Position& operator+=(const Position& pos)
{
*this = *this + pos;
return *this;
}
#define RELATIONAL_OPERATOR(op) \
bool operator op(const Position& rhs) const \
{ \
return std::tie(minute, second, frame) op std::tie(rhs.minute, rhs.second, rhs.frame); \
}
RELATIONAL_OPERATOR(==);
RELATIONAL_OPERATOR(!=);
RELATIONAL_OPERATOR(<);
RELATIONAL_OPERATOR(<=);
RELATIONAL_OPERATOR(>);
RELATIONAL_OPERATOR(>=);
#undef RELATIONAL_OPERATOR
};
union SubChannelQ
{
union Control
{
u8 bits;
BitField<u8, u8, 0, 4> adr;
BitField<u8, bool, 4, 1> audio_preemphasis;
BitField<u8, bool, 5, 1> digital_copy_permitted;
BitField<u8, bool, 6, 1> data;
BitField<u8, bool, 7, 1> four_channel_audio;
};
struct
{
Control control;
u8 track_number_bcd;
u8 index_number_bcd;
u8 relative_minute_bcd;
u8 relative_second_bcd;
u8 relative_frame_bcd;
u8 reserved;
u8 absolute_minute_bcd;
u8 absolute_second_bcd;
u8 absolute_frame_bcd;
u16 crc;
};
u8 data[SUBCHANNEL_BYTES_PER_FRAME];
static u16 ComputeCRC(const u8* data);
bool IsCRCValid() const;
SubChannelQ& operator=(const SubChannelQ& q)
{
std::copy(q.data, q.data + SUBCHANNEL_BYTES_PER_FRAME, data);
return *this;
}
};
static_assert(sizeof(SubChannelQ) == SUBCHANNEL_BYTES_PER_FRAME, "SubChannelQ is correct size");
// Helper functions.
static u32 GetBytesPerSector(TrackMode mode);
// Opening disc image.
static std::unique_ptr<CDImage> Open(const char* filename);
static std::unique_ptr<CDImage> OpenBinImage(const char* filename);
static std::unique_ptr<CDImage> OpenCueSheetImage(const char* filename);
2020-01-30 05:50:14 +00:00
static std::unique_ptr<CDImage> OpenCHDImage(const char* filename);
2019-09-20 10:14:00 +00:00
// Accessors.
2019-10-17 13:54:51 +00:00
const std::string& GetFileName() const { return m_filename; }
LBA GetPositionOnDisc() const { return m_position_on_disc; }
Position GetMSFPositionOnDisc() const { return Position::FromLBA(m_position_on_disc); }
LBA GetPositionInTrack() const { return m_position_in_track; }
Position GetMSFPositionInTrack() const { return Position::FromLBA(m_position_in_track); }
LBA GetLBACount() const { return m_lba_count; }
2019-10-18 12:44:28 +00:00
u32 GetIndexNumber() const { return m_current_index->index_number; }
u32 GetTrackNumber() const { return m_current_index->track_number; }
2019-10-18 12:44:28 +00:00
u32 GetTrackCount() const { return static_cast<u32>(m_tracks.size()); }
2019-10-18 13:06:13 +00:00
LBA GetTrackStartPosition(u8 track) const;
Position GetTrackStartMSFPosition(u8 track) const;
2020-05-16 10:00:53 +00:00
LBA GetTrackLength(u8 track) const;
Position GetTrackMSFLength(u8 track) const;
TrackMode GetTrackMode(u8 track) const;
u32 GetFirstTrackNumber() const { return m_tracks.front().track_number; }
u32 GetLastTrackNumber() const { return m_tracks.back().track_number; }
2019-09-20 10:14:00 +00:00
// Seek to data LBA.
bool Seek(LBA lba);
2019-09-20 10:14:00 +00:00
// Seek to disc position (MSF).
bool Seek(const Position& pos);
2019-09-20 10:14:00 +00:00
// Seek to track and position.
bool Seek(u32 track_number, const Position& pos_in_track);
2019-09-20 10:14:00 +00:00
2019-11-29 13:36:25 +00:00
// Seek to track and LBA.
bool Seek(u32 track_number, LBA lba);
2019-09-20 10:14:00 +00:00
// Read from the current LBA. Returns the number of sectors read.
u32 Read(ReadMode read_mode, u32 sector_count, void* buffer);
2019-10-27 03:34:13 +00:00
// Read a single raw sector from the current LBA.
bool ReadRawSector(void* buffer);
// Reads sub-channel Q for the current LBA.
virtual bool ReadSubChannelQ(SubChannelQ* subq);
protected:
struct Track
{
u32 track_number;
LBA start_lba;
u32 first_index;
u32 length;
TrackMode mode;
SubChannelQ::Control control;
};
struct Index
{
u64 file_offset;
u32 file_index;
u32 file_sector_size;
LBA start_lba_on_disc;
u32 track_number;
2019-10-18 12:44:28 +00:00
u32 index_number;
LBA start_lba_in_track;
u32 length;
TrackMode mode;
SubChannelQ::Control control;
bool is_pregap;
};
// Reads a single sector from an index.
virtual bool ReadSectorFromIndex(void* buffer, const Index& index, LBA lba_in_index) = 0;
const Index* GetIndexForDiscPosition(LBA pos);
const Index* GetIndexForTrackPosition(u32 track_number, LBA track_pos);
/// Generates sub-channel Q given the specified position.
bool GenerateSubChannelQ(SubChannelQ* subq, LBA lba);
/// Generates sub-channel Q from the given index and index-offset.
void GenerateSubChannelQ(SubChannelQ* subq, const Index* index, u32 index_offset);
/// Synthesis of lead-out data.
void AddLeadOutIndex();
2019-10-17 13:54:51 +00:00
std::string m_filename;
u32 m_lba_count = 0;
2019-10-17 13:54:51 +00:00
std::vector<Track> m_tracks;
std::vector<Index> m_indices;
2019-09-20 10:14:00 +00:00
// Position on disc.
LBA m_position_on_disc = 0;
// Position in track/index.
const Index* m_current_index = nullptr;
LBA m_position_in_index = 0;
LBA m_position_in_track = 0;
2019-09-20 10:14:00 +00:00
};