// SPDX-FileCopyrightText: 2019-2023 Connor McLaughlin // SPDX-License-Identifier: (GPL-3.0 OR CC-BY-NC-ND-4.0) #include "cd_image.h" #include "cd_subchannel_replacement.h" #include "common/error.h" #include "common/file_system.h" #include "common/log.h" #include Log_SetChannel(CDImageBin); namespace { class CDImageBin : public CDImage { public: CDImageBin(); ~CDImageBin() override; bool Open(const char* filename, Error* error); bool ReadSubChannelQ(SubChannelQ* subq, const Index& index, LBA lba_in_index) override; bool HasNonStandardSubchannel() const override; protected: bool ReadSectorFromIndex(void* buffer, const Index& index, LBA lba_in_index) override; private: std::FILE* m_fp = nullptr; u64 m_file_position = 0; CDSubChannelReplacement m_sbi; }; } // namespace CDImageBin::CDImageBin() = default; CDImageBin::~CDImageBin() { if (m_fp) std::fclose(m_fp); } bool CDImageBin::Open(const char* filename, Error* error) { m_filename = filename; m_fp = FileSystem::OpenCFile(filename, "rb"); if (!m_fp) { Log_ErrorPrintf("Failed to open binfile '%s': errno %d", filename, errno); if (error) error->SetErrno(errno); 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(std::ftell(m_fp)); std::fseek(m_fp, 0, SEEK_SET); m_lba_count = file_size / track_sector_size; SubChannelQ::Control control = {}; TrackMode mode = TrackMode::Mode2Raw; control.data = mode != TrackMode::Audio; // Two seconds default pregap. const u32 pregap_frames = 2 * FRAMES_PER_SECOND; Index pregap_index = {}; pregap_index.file_sector_size = track_sector_size; pregap_index.start_lba_on_disc = 0; pregap_index.start_lba_in_track = static_cast(-static_cast(pregap_frames)); pregap_index.length = pregap_frames; pregap_index.track_number = 1; pregap_index.index_number = 0; pregap_index.mode = mode; pregap_index.submode = CDImage::SubchannelMode::None; pregap_index.control.bits = control.bits; pregap_index.is_pregap = true; m_indices.push_back(pregap_index); // Data index. Index data_index = {}; data_index.file_index = 0; 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; data_index.index_number = 1; data_index.start_lba_in_track = 0; data_index.length = m_lba_count; data_index.mode = mode; data_index.submode = CDImage::SubchannelMode::None; data_index.control.bits = control.bits; m_indices.push_back(data_index); // Assume a single track. m_tracks.push_back(Track{static_cast(1), data_index.start_lba_on_disc, static_cast(0), m_lba_count, mode, SubchannelMode::None, control}); AddLeadOutIndex(); m_sbi.LoadFromImagePath(filename); return Seek(1, Position{0, 0, 0}); } bool CDImageBin::ReadSubChannelQ(SubChannelQ* subq, const Index& index, LBA lba_in_index) { if (m_sbi.GetReplacementSubChannelQ(index.start_lba_on_disc + lba_in_index, subq)) return true; return CDImage::ReadSubChannelQ(subq, index, lba_in_index); } bool CDImageBin::HasNonStandardSubchannel() const { return (m_sbi.GetReplacementSectorCount() > 0); } bool CDImageBin::ReadSectorFromIndex(void* buffer, const Index& index, LBA lba_in_index) { const u64 file_position = index.file_offset + (static_cast(lba_in_index) * index.file_sector_size); if (m_file_position != file_position) { if (std::fseek(m_fp, static_cast(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(m_file_position), SEEK_SET); return false; } m_file_position += index.file_sector_size; return true; } std::unique_ptr CDImage::OpenBinImage(const char* filename, Error* error) { std::unique_ptr image = std::make_unique(); if (!image->Open(filename, error)) return {}; return image; }