2022-12-04 11:03:45 +00:00
|
|
|
// SPDX-FileCopyrightText: 2019-2022 Connor McLaughlin <stenzek@gmail.com>
|
|
|
|
// SPDX-License-Identifier: (GPL-3.0 OR CC-BY-NC-ND-4.0)
|
|
|
|
|
2019-09-20 10:14:00 +00:00
|
|
|
#pragma once
|
2022-07-08 12:43:38 +00:00
|
|
|
#include "common/bitfield.h"
|
|
|
|
#include "common/progress_callback.h"
|
|
|
|
#include "common/types.h"
|
2020-06-23 09:04:16 +00:00
|
|
|
#include <array>
|
2019-10-18 08:18:04 +00:00
|
|
|
#include <memory>
|
2019-11-27 15:55:09 +00:00
|
|
|
#include <string>
|
2019-10-17 13:54:51 +00:00
|
|
|
#include <tuple>
|
2019-10-18 08:18:04 +00:00
|
|
|
#include <vector>
|
2019-09-20 10:14:00 +00:00
|
|
|
|
2021-03-18 15:52:00 +00:00
|
|
|
class Error;
|
|
|
|
|
2019-09-20 10:14:00 +00:00
|
|
|
class CDImage
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
CDImage();
|
2019-10-18 08:18:04 +00:00
|
|
|
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,
|
2019-10-04 09:05:19 +00:00
|
|
|
SECTOR_HEADER_SIZE = 4,
|
2019-11-10 12:44:53 +00:00
|
|
|
FRAMES_PER_SECOND = 75, // "sectors", or "timecode frames" (not "channel frames")
|
2019-09-21 15:12:16 +00:00
|
|
|
SECONDS_PER_MINUTE = 60,
|
2019-11-10 12:44:53 +00:00
|
|
|
FRAMES_PER_MINUTE = FRAMES_PER_SECOND * SECONDS_PER_MINUTE,
|
2020-05-08 00:50:22 +00:00
|
|
|
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
|
|
|
|
{
|
2019-10-04 09:05:19 +00:00
|
|
|
DataOnly, // 2048 bytes per sector.
|
|
|
|
RawSector, // 2352 bytes per sector.
|
|
|
|
RawNoSync, // 2340 bytes per sector.
|
|
|
|
};
|
|
|
|
|
2019-11-10 12:44:53 +00:00
|
|
|
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
|
|
|
|
};
|
|
|
|
|
2022-04-03 10:55:27 +00:00
|
|
|
enum class PrecacheResult : u8
|
|
|
|
{
|
|
|
|
Unsupported,
|
|
|
|
ReadError,
|
|
|
|
Success,
|
|
|
|
};
|
|
|
|
|
2019-10-04 09:05:19 +00:00
|
|
|
struct SectorHeader
|
|
|
|
{
|
|
|
|
u8 minute;
|
|
|
|
u8 second;
|
|
|
|
u8 frame;
|
|
|
|
u8 sector_mode;
|
|
|
|
};
|
|
|
|
|
2019-10-18 08:18:04 +00:00
|
|
|
struct Position
|
2019-10-17 13:54:51 +00:00
|
|
|
{
|
2019-10-18 08:18:04 +00:00
|
|
|
u8 minute;
|
|
|
|
u8 second;
|
|
|
|
u8 frame;
|
|
|
|
|
|
|
|
static constexpr Position FromBCD(u8 minute, u8 second, u8 frame)
|
|
|
|
{
|
2019-12-06 06:23:08 +00:00
|
|
|
return Position{PackedBCDToBinary(minute), PackedBCDToBinary(second), PackedBCDToBinary(frame)};
|
2019-10-18 08:18:04 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
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);
|
|
|
|
}
|
|
|
|
|
2019-11-10 12:44:53 +00:00
|
|
|
constexpr std::tuple<u8, u8, u8> ToBCD() const
|
|
|
|
{
|
2019-12-06 06:23:08 +00:00
|
|
|
return std::make_tuple<u8, u8, u8>(BinaryToBCD(minute), BinaryToBCD(second), BinaryToBCD(frame));
|
2019-11-10 12:44:53 +00:00
|
|
|
}
|
|
|
|
|
2019-10-18 08:18:04 +00:00
|
|
|
Position operator+(const Position& rhs) { return FromLBA(ToLBA() + rhs.ToLBA()); }
|
|
|
|
Position& operator+=(const Position& pos)
|
|
|
|
{
|
|
|
|
*this = *this + pos;
|
|
|
|
return *this;
|
|
|
|
}
|
2019-10-26 14:01:25 +00:00
|
|
|
|
|
|
|
#define RELATIONAL_OPERATOR(op) \
|
2019-11-10 12:44:53 +00:00
|
|
|
bool operator op(const Position& rhs) const \
|
2019-10-26 14:01:25 +00:00
|
|
|
{ \
|
|
|
|
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
|
2019-10-18 08:18:04 +00:00
|
|
|
};
|
|
|
|
|
2019-11-10 12:44:53 +00:00
|
|
|
union SubChannelQ
|
|
|
|
{
|
2020-06-23 09:04:16 +00:00
|
|
|
using Data = std::array<u8, SUBCHANNEL_BYTES_PER_FRAME>;
|
|
|
|
|
2019-11-10 12:44:53 +00:00
|
|
|
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;
|
2021-06-01 11:42:50 +00:00
|
|
|
|
2023-09-03 04:30:26 +00:00
|
|
|
Control() = default;
|
|
|
|
|
|
|
|
Control(u8 bits_) : bits(bits_) {}
|
|
|
|
|
|
|
|
Control(const Control& rhs) : bits(rhs.bits) {}
|
|
|
|
|
2021-06-01 11:42:50 +00:00
|
|
|
Control& operator=(const Control& rhs)
|
|
|
|
{
|
|
|
|
bits = rhs.bits;
|
|
|
|
return *this;
|
|
|
|
}
|
2019-11-10 12:44:53 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
struct
|
|
|
|
{
|
2021-06-01 12:04:25 +00:00
|
|
|
u8 control_bits;
|
2019-11-10 12:44:53 +00:00
|
|
|
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;
|
|
|
|
};
|
|
|
|
|
2020-06-23 09:04:16 +00:00
|
|
|
Data data;
|
2019-11-10 12:44:53 +00:00
|
|
|
|
2020-06-23 09:04:16 +00:00
|
|
|
static u16 ComputeCRC(const Data& data);
|
2019-12-06 06:23:08 +00:00
|
|
|
|
2023-09-03 04:30:26 +00:00
|
|
|
Control GetControl() const { return Control(control_bits); }
|
2021-06-01 12:04:25 +00:00
|
|
|
bool IsData() const { return GetControl().data; }
|
|
|
|
|
2019-12-06 06:23:08 +00:00
|
|
|
bool IsCRCValid() const;
|
2019-11-12 10:31:59 +00:00
|
|
|
|
2023-09-03 04:30:26 +00:00
|
|
|
SubChannelQ() = default;
|
|
|
|
|
|
|
|
SubChannelQ(const SubChannelQ& q) : data(q.data) {}
|
|
|
|
|
2019-11-12 10:31:59 +00:00
|
|
|
SubChannelQ& operator=(const SubChannelQ& q)
|
|
|
|
{
|
2020-06-23 09:04:16 +00:00
|
|
|
data = q.data;
|
2019-11-12 10:31:59 +00:00
|
|
|
return *this;
|
|
|
|
}
|
2019-11-10 12:44:53 +00:00
|
|
|
};
|
|
|
|
static_assert(sizeof(SubChannelQ) == SUBCHANNEL_BYTES_PER_FRAME, "SubChannelQ is correct size");
|
|
|
|
|
2020-07-21 14:03:07 +00:00
|
|
|
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;
|
|
|
|
u32 index_number;
|
|
|
|
LBA start_lba_in_track;
|
|
|
|
u32 length;
|
|
|
|
TrackMode mode;
|
|
|
|
SubChannelQ::Control control;
|
|
|
|
bool is_pregap;
|
|
|
|
};
|
|
|
|
|
2019-11-10 12:44:53 +00:00
|
|
|
// Helper functions.
|
|
|
|
static u32 GetBytesPerSector(TrackMode mode);
|
|
|
|
|
2021-07-24 08:38:54 +00:00
|
|
|
/// Returns a list of physical CD-ROM devices, .first being the device path, .second being the device name.
|
|
|
|
static std::vector<std::pair<std::string, std::string>> GetDeviceList();
|
|
|
|
|
|
|
|
/// Returns true if the specified filename is a CD-ROM device name.
|
|
|
|
static bool IsDeviceName(const char* filename);
|
|
|
|
|
2019-10-18 08:18:04 +00:00
|
|
|
// Opening disc image.
|
2023-08-19 13:40:36 +00:00
|
|
|
static std::unique_ptr<CDImage> Open(const char* filename, bool allow_patches, Error* error);
|
|
|
|
static std::unique_ptr<CDImage> OpenBinImage(const char* filename, Error* error);
|
|
|
|
static std::unique_ptr<CDImage> OpenCueSheetImage(const char* filename, Error* error);
|
|
|
|
static std::unique_ptr<CDImage> OpenCHDImage(const char* filename, Error* error);
|
|
|
|
static std::unique_ptr<CDImage> OpenEcmImage(const char* filename, Error* error);
|
|
|
|
static std::unique_ptr<CDImage> OpenMdsImage(const char* filename, Error* error);
|
|
|
|
static std::unique_ptr<CDImage> OpenPBPImage(const char* filename, Error* error);
|
|
|
|
static std::unique_ptr<CDImage> OpenM3uImage(const char* filename, bool apply_patches, Error* error);
|
|
|
|
static std::unique_ptr<CDImage> OpenDeviceImage(const char* filename, Error* error);
|
2020-07-21 14:03:07 +00:00
|
|
|
static std::unique_ptr<CDImage>
|
|
|
|
CreateMemoryImage(CDImage* image, ProgressCallback* progress = ProgressCallback::NullProgressCallback);
|
2021-06-01 11:42:50 +00:00
|
|
|
static std::unique_ptr<CDImage> OverlayPPFPatch(const char* filename, std::unique_ptr<CDImage> parent_image,
|
|
|
|
ProgressCallback* progress = ProgressCallback::NullProgressCallback);
|
2019-09-20 10:14:00 +00:00
|
|
|
|
|
|
|
// Accessors.
|
2023-09-03 04:30:26 +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; }
|
|
|
|
u32 GetIndexNumber() const { return m_current_index->index_number; }
|
|
|
|
u32 GetTrackNumber() const { return m_current_index->track_number; }
|
|
|
|
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;
|
2020-06-07 15:19:35 +00:00
|
|
|
LBA GetTrackIndexPosition(u8 track, u8 index) const;
|
|
|
|
LBA GetTrackIndexLength(u8 track, u8 index) const;
|
2023-09-03 04:30:26 +00:00
|
|
|
u32 GetFirstTrackNumber() const { return m_tracks.front().track_number; }
|
|
|
|
u32 GetLastTrackNumber() const { return m_tracks.back().track_number; }
|
|
|
|
u32 GetIndexCount() const { return static_cast<u32>(m_indices.size()); }
|
|
|
|
const std::vector<Track>& GetTracks() const { return m_tracks; }
|
|
|
|
const std::vector<Index>& GetIndices() const { return m_indices; }
|
2020-07-21 14:03:07 +00:00
|
|
|
const Track& GetTrack(u32 track) const;
|
|
|
|
const Index& GetIndex(u32 i) const;
|
2019-09-20 10:14:00 +00:00
|
|
|
|
|
|
|
// Seek to data LBA.
|
2019-10-18 08:18:04 +00:00
|
|
|
bool Seek(LBA lba);
|
2019-09-20 10:14:00 +00:00
|
|
|
|
2019-10-18 08:18:04 +00:00
|
|
|
// Seek to disc position (MSF).
|
|
|
|
bool Seek(const Position& pos);
|
2019-09-20 10:14:00 +00:00
|
|
|
|
2019-10-18 08:18:04 +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);
|
|
|
|
|
2021-05-26 17:46:33 +00:00
|
|
|
// Read a single raw sector, and subchannel from the current LBA.
|
|
|
|
bool ReadRawSector(void* buffer, SubChannelQ* subq);
|
2021-03-26 16:19:23 +00:00
|
|
|
|
|
|
|
// Reads sub-channel Q for the specified index+LBA.
|
|
|
|
virtual bool ReadSubChannelQ(SubChannelQ* subq, const Index& index, LBA lba_in_index);
|
2019-11-10 12:44:53 +00:00
|
|
|
|
2020-12-17 17:01:57 +00:00
|
|
|
// Returns true if the image has replacement subchannel data.
|
|
|
|
virtual bool HasNonStandardSubchannel() const;
|
|
|
|
|
2020-01-30 05:50:00 +00:00
|
|
|
// Reads a single sector from an index.
|
|
|
|
virtual bool ReadSectorFromIndex(void* buffer, const Index& index, LBA lba_in_index) = 0;
|
|
|
|
|
2021-03-26 16:19:23 +00:00
|
|
|
// Retrieve image metadata.
|
|
|
|
virtual std::string GetMetadata(const std::string_view& type) const;
|
|
|
|
|
|
|
|
// Returns true if this image type has sub-images (e.g. m3u).
|
|
|
|
virtual bool HasSubImages() const;
|
|
|
|
|
|
|
|
// Returns the number of sub-images in this image, if the format supports multiple.
|
|
|
|
virtual u32 GetSubImageCount() const;
|
|
|
|
|
|
|
|
// Returns the current sub-image index, if any.
|
|
|
|
virtual u32 GetCurrentSubImage() const;
|
|
|
|
|
|
|
|
// Changes the current sub-image. If this fails, the image state is unchanged.
|
2023-08-19 13:40:36 +00:00
|
|
|
virtual bool SwitchSubImage(u32 index, Error* error);
|
2021-03-26 16:19:23 +00:00
|
|
|
|
|
|
|
// Retrieve sub-image metadata.
|
|
|
|
virtual std::string GetSubImageMetadata(u32 index, const std::string_view& type) const;
|
|
|
|
|
2022-04-03 10:55:27 +00:00
|
|
|
// Returns true if the source supports precaching, which may be more optimal than an in-memory copy.
|
|
|
|
virtual PrecacheResult Precache(ProgressCallback* progress = ProgressCallback::NullProgressCallback);
|
2022-07-23 03:23:54 +00:00
|
|
|
virtual bool IsPrecached() const;
|
2022-04-03 10:55:27 +00:00
|
|
|
|
2020-07-21 14:03:07 +00:00
|
|
|
protected:
|
2021-03-26 17:34:50 +00:00
|
|
|
void ClearTOC();
|
2021-03-26 16:19:23 +00:00
|
|
|
void CopyTOC(const CDImage* image);
|
|
|
|
|
2019-10-18 08:18:04 +00:00
|
|
|
const Index* GetIndexForDiscPosition(LBA pos);
|
|
|
|
const Index* GetIndexForTrackPosition(u32 track_number, LBA track_pos);
|
|
|
|
|
2019-11-10 12:44:53 +00:00
|
|
|
/// 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.
|
2021-03-26 16:19:23 +00:00
|
|
|
void GenerateSubChannelQ(SubChannelQ* subq, const Index& index, u32 index_offset);
|
2019-11-10 12:44:53 +00:00
|
|
|
|
2020-05-08 00:50:22 +00:00
|
|
|
/// Synthesis of lead-out data.
|
|
|
|
void AddLeadOutIndex();
|
|
|
|
|
2019-10-17 13:54:51 +00:00
|
|
|
std::string m_filename;
|
2019-10-18 08:18:04 +00:00
|
|
|
u32 m_lba_count = 0;
|
2019-10-17 13:54:51 +00:00
|
|
|
|
2019-10-18 08:18:04 +00:00
|
|
|
std::vector<Track> m_tracks;
|
|
|
|
std::vector<Index> m_indices;
|
2019-09-20 10:14:00 +00:00
|
|
|
|
2021-03-26 16:19:23 +00:00
|
|
|
private:
|
2019-10-18 08:18:04 +00:00
|
|
|
// Position on disc.
|
|
|
|
LBA m_position_on_disc = 0;
|
2019-09-25 14:15:06 +00:00
|
|
|
|
2019-10-18 08:18:04 +00:00
|
|
|
// 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
|
|
|
};
|