2023-11-18 06:21:51 +00:00
|
|
|
// SPDX-FileCopyrightText: 2019-2023 Connor McLaughlin <stenzek@gmail.com>
|
2022-12-04 11:03:45 +00:00
|
|
|
// SPDX-License-Identifier: (GPL-3.0 OR CC-BY-NC-ND-4.0)
|
|
|
|
|
2019-10-18 08:18:04 +00:00
|
|
|
#include "cd_image.h"
|
2019-12-06 06:23:08 +00:00
|
|
|
#include "cd_subchannel_replacement.h"
|
2023-11-18 06:21:51 +00:00
|
|
|
|
2022-07-08 12:43:38 +00:00
|
|
|
#include "common/error.h"
|
|
|
|
#include "common/file_system.h"
|
|
|
|
#include "common/log.h"
|
2023-11-18 06:21:51 +00:00
|
|
|
|
2020-05-06 13:42:04 +00:00
|
|
|
#include <cerrno>
|
2023-11-18 06:21:51 +00:00
|
|
|
|
2019-10-18 08:18:04 +00:00
|
|
|
Log_SetChannel(CDImageBin);
|
|
|
|
|
2023-11-18 06:21:51 +00:00
|
|
|
namespace {
|
|
|
|
|
2019-10-18 08:18:04 +00:00
|
|
|
class CDImageBin : public CDImage
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
CDImageBin();
|
|
|
|
~CDImageBin() override;
|
|
|
|
|
2023-08-19 13:40:36 +00:00
|
|
|
bool Open(const char* filename, Error* error);
|
2019-10-18 08:18:04 +00:00
|
|
|
|
2021-03-26 16:19:23 +00:00
|
|
|
bool ReadSubChannelQ(SubChannelQ* subq, const Index& index, LBA lba_in_index) override;
|
2020-12-17 17:01:57 +00:00
|
|
|
bool HasNonStandardSubchannel() const override;
|
2019-12-06 06:23:08 +00:00
|
|
|
|
2023-12-20 13:40:24 +00:00
|
|
|
s64 GetSizeOnDisk() const override;
|
|
|
|
|
2020-01-30 05:50:00 +00:00
|
|
|
protected:
|
|
|
|
bool ReadSectorFromIndex(void* buffer, const Index& index, LBA lba_in_index) override;
|
|
|
|
|
2019-10-18 08:18:04 +00:00
|
|
|
private:
|
|
|
|
std::FILE* m_fp = nullptr;
|
2020-01-30 05:50:00 +00:00
|
|
|
u64 m_file_position = 0;
|
2019-12-06 06:23:08 +00:00
|
|
|
|
|
|
|
CDSubChannelReplacement m_sbi;
|
2019-10-18 08:18:04 +00:00
|
|
|
};
|
|
|
|
|
2023-11-18 06:21:51 +00:00
|
|
|
} // namespace
|
|
|
|
|
2019-10-18 08:18:04 +00:00
|
|
|
CDImageBin::CDImageBin() = default;
|
|
|
|
|
|
|
|
CDImageBin::~CDImageBin()
|
|
|
|
{
|
|
|
|
if (m_fp)
|
|
|
|
std::fclose(m_fp);
|
|
|
|
}
|
|
|
|
|
2023-08-19 13:40:36 +00:00
|
|
|
bool CDImageBin::Open(const char* filename, Error* error)
|
2019-10-18 08:18:04 +00:00
|
|
|
{
|
2019-10-18 12:04:25 +00:00
|
|
|
m_filename = filename;
|
2020-01-10 03:31:12 +00:00
|
|
|
m_fp = FileSystem::OpenCFile(filename, "rb");
|
2019-10-18 08:18:04 +00:00
|
|
|
if (!m_fp)
|
|
|
|
{
|
2020-05-06 13:42:04 +00:00
|
|
|
Log_ErrorPrintf("Failed to open binfile '%s': errno %d", filename, errno);
|
2021-03-18 15:52:00 +00:00
|
|
|
if (error)
|
|
|
|
error->SetErrno(errno);
|
2019-10-18 08:18:04 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
const u32 track_sector_size = RAW_SECTOR_SIZE;
|
|
|
|
|
|
|
|
// determine the length from the file
|
|
|
|
std::fseek(m_fp, 0, SEEK_END);
|
|
|
|
const u32 file_size = static_cast<u32>(std::ftell(m_fp));
|
|
|
|
std::fseek(m_fp, 0, SEEK_SET);
|
|
|
|
|
|
|
|
m_lba_count = file_size / track_sector_size;
|
|
|
|
|
2019-11-10 12:44:53 +00:00
|
|
|
SubChannelQ::Control control = {};
|
|
|
|
TrackMode mode = TrackMode::Mode2Raw;
|
2019-11-13 06:28:43 +00:00
|
|
|
control.data = mode != TrackMode::Audio;
|
2019-11-10 12:44:53 +00:00
|
|
|
|
2019-10-18 08:18:04 +00:00
|
|
|
// Two seconds default pregap.
|
|
|
|
const u32 pregap_frames = 2 * FRAMES_PER_SECOND;
|
|
|
|
Index pregap_index = {};
|
2019-11-10 12:44:53 +00:00
|
|
|
pregap_index.file_sector_size = track_sector_size;
|
2019-10-18 08:18:04 +00:00
|
|
|
pregap_index.start_lba_on_disc = 0;
|
|
|
|
pregap_index.start_lba_in_track = static_cast<LBA>(-static_cast<s32>(pregap_frames));
|
|
|
|
pregap_index.length = pregap_frames;
|
|
|
|
pregap_index.track_number = 1;
|
2019-10-18 12:44:28 +00:00
|
|
|
pregap_index.index_number = 0;
|
2019-11-10 12:44:53 +00:00
|
|
|
pregap_index.mode = mode;
|
2023-11-05 10:36:28 +00:00
|
|
|
pregap_index.submode = CDImage::SubchannelMode::None;
|
2019-11-10 12:44:53 +00:00
|
|
|
pregap_index.control.bits = control.bits;
|
2019-10-18 08:18:04 +00:00
|
|
|
pregap_index.is_pregap = true;
|
|
|
|
m_indices.push_back(pregap_index);
|
|
|
|
|
|
|
|
// Data index.
|
|
|
|
Index data_index = {};
|
2020-01-30 05:50:00 +00:00
|
|
|
data_index.file_index = 0;
|
2019-10-18 08:18:04 +00:00
|
|
|
data_index.file_offset = 0;
|
|
|
|
data_index.file_sector_size = track_sector_size;
|
|
|
|
data_index.start_lba_on_disc = pregap_index.length;
|
|
|
|
data_index.track_number = 1;
|
2019-10-18 12:44:28 +00:00
|
|
|
data_index.index_number = 1;
|
2019-10-18 08:18:04 +00:00
|
|
|
data_index.start_lba_in_track = 0;
|
|
|
|
data_index.length = m_lba_count;
|
2019-11-10 12:44:53 +00:00
|
|
|
data_index.mode = mode;
|
2023-11-05 10:36:28 +00:00
|
|
|
data_index.submode = CDImage::SubchannelMode::None;
|
2019-11-10 12:44:53 +00:00
|
|
|
data_index.control.bits = control.bits;
|
2019-10-18 08:18:04 +00:00
|
|
|
m_indices.push_back(data_index);
|
|
|
|
|
|
|
|
// Assume a single track.
|
2023-11-05 10:36:28 +00:00
|
|
|
m_tracks.push_back(Track{static_cast<u32>(1), data_index.start_lba_on_disc, static_cast<u32>(0), m_lba_count, mode,
|
|
|
|
SubchannelMode::None, control});
|
2019-10-18 08:18:04 +00:00
|
|
|
|
2020-05-08 00:50:22 +00:00
|
|
|
AddLeadOutIndex();
|
|
|
|
|
2023-11-15 09:05:21 +00:00
|
|
|
m_sbi.LoadFromImagePath(filename);
|
2019-12-06 06:23:08 +00:00
|
|
|
|
2019-10-18 08:18:04 +00:00
|
|
|
return Seek(1, Position{0, 0, 0});
|
|
|
|
}
|
|
|
|
|
2021-03-26 16:19:23 +00:00
|
|
|
bool CDImageBin::ReadSubChannelQ(SubChannelQ* subq, const Index& index, LBA lba_in_index)
|
2019-12-06 06:23:08 +00:00
|
|
|
{
|
2021-03-26 16:19:23 +00:00
|
|
|
if (m_sbi.GetReplacementSubChannelQ(index.start_lba_on_disc + lba_in_index, subq))
|
2019-12-06 06:23:08 +00:00
|
|
|
return true;
|
|
|
|
|
2021-03-26 16:19:23 +00:00
|
|
|
return CDImage::ReadSubChannelQ(subq, index, lba_in_index);
|
2019-12-06 06:23:08 +00:00
|
|
|
}
|
|
|
|
|
2020-12-17 17:01:57 +00:00
|
|
|
bool CDImageBin::HasNonStandardSubchannel() const
|
|
|
|
{
|
|
|
|
return (m_sbi.GetReplacementSectorCount() > 0);
|
|
|
|
}
|
|
|
|
|
2020-01-30 05:50:00 +00:00
|
|
|
bool CDImageBin::ReadSectorFromIndex(void* buffer, const Index& index, LBA lba_in_index)
|
|
|
|
{
|
|
|
|
const u64 file_position = index.file_offset + (static_cast<u64>(lba_in_index) * index.file_sector_size);
|
|
|
|
if (m_file_position != file_position)
|
|
|
|
{
|
|
|
|
if (std::fseek(m_fp, static_cast<long>(file_position), SEEK_SET) != 0)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
m_file_position = file_position;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (std::fread(buffer, index.file_sector_size, 1, m_fp) != 1)
|
|
|
|
{
|
|
|
|
std::fseek(m_fp, static_cast<long>(m_file_position), SEEK_SET);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
m_file_position += index.file_sector_size;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2023-12-20 13:40:24 +00:00
|
|
|
s64 CDImageBin::GetSizeOnDisk() const
|
|
|
|
{
|
|
|
|
return FileSystem::FSize64(m_fp);
|
|
|
|
}
|
|
|
|
|
2023-08-19 13:40:36 +00:00
|
|
|
std::unique_ptr<CDImage> CDImage::OpenBinImage(const char* filename, Error* error)
|
2019-10-18 08:18:04 +00:00
|
|
|
{
|
|
|
|
std::unique_ptr<CDImageBin> image = std::make_unique<CDImageBin>();
|
2021-03-18 15:52:00 +00:00
|
|
|
if (!image->Open(filename, error))
|
2019-10-18 08:18:04 +00:00
|
|
|
return {};
|
|
|
|
|
|
|
|
return image;
|
|
|
|
}
|