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)
|
|
|
|
|
2020-07-21 14:03:07 +00:00
|
|
|
#include "cd_image.h"
|
|
|
|
#include "cd_subchannel_replacement.h"
|
2023-11-18 06:21:51 +00:00
|
|
|
|
2022-07-08 12:43:38 +00:00
|
|
|
#include "common/assert.h"
|
|
|
|
#include "common/file_system.h"
|
|
|
|
#include "common/log.h"
|
|
|
|
#include "common/path.h"
|
2023-11-18 06:21:51 +00:00
|
|
|
|
2020-07-21 14:03:07 +00:00
|
|
|
#include <algorithm>
|
|
|
|
#include <cerrno>
|
2023-11-18 06:21:51 +00:00
|
|
|
|
2020-07-21 14:03:07 +00:00
|
|
|
Log_SetChannel(CDImageMemory);
|
|
|
|
|
2023-11-18 06:21:51 +00:00
|
|
|
namespace {
|
|
|
|
|
2020-07-21 14:03:07 +00:00
|
|
|
class CDImageMemory : public CDImage
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
CDImageMemory();
|
|
|
|
~CDImageMemory() override;
|
|
|
|
|
|
|
|
bool CopyImage(CDImage* image, ProgressCallback* progress);
|
|
|
|
|
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;
|
2020-07-21 14:03:07 +00:00
|
|
|
|
2022-07-23 03:23:54 +00:00
|
|
|
bool IsPrecached() const override;
|
|
|
|
|
2020-07-21 14:03:07 +00:00
|
|
|
protected:
|
|
|
|
bool ReadSectorFromIndex(void* buffer, const Index& index, LBA lba_in_index) override;
|
|
|
|
|
|
|
|
private:
|
|
|
|
u8* m_memory = nullptr;
|
|
|
|
u32 m_memory_sectors = 0;
|
|
|
|
CDSubChannelReplacement m_sbi;
|
|
|
|
};
|
|
|
|
|
2023-11-18 06:21:51 +00:00
|
|
|
} // namespace
|
|
|
|
|
2020-07-21 14:03:07 +00:00
|
|
|
CDImageMemory::CDImageMemory() = default;
|
|
|
|
|
|
|
|
CDImageMemory::~CDImageMemory()
|
|
|
|
{
|
|
|
|
if (m_memory)
|
|
|
|
std::free(m_memory);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool CDImageMemory::CopyImage(CDImage* image, ProgressCallback* progress)
|
|
|
|
{
|
|
|
|
// figure out the total number of sectors (not including blank pregaps)
|
|
|
|
m_memory_sectors = 0;
|
|
|
|
for (u32 i = 0; i < image->GetIndexCount(); i++)
|
|
|
|
{
|
|
|
|
const Index& index = image->GetIndex(i);
|
|
|
|
if (index.file_sector_size > 0)
|
|
|
|
m_memory_sectors += image->GetIndex(i).length;
|
|
|
|
}
|
|
|
|
|
|
|
|
if ((static_cast<u64>(RAW_SECTOR_SIZE) * static_cast<u64>(m_memory_sectors)) >=
|
|
|
|
static_cast<u64>(std::numeric_limits<size_t>::max()))
|
|
|
|
{
|
|
|
|
progress->DisplayFormattedModalError("Insufficient address space");
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
progress->SetFormattedStatusText("Allocating memory for %u sectors...", m_memory_sectors);
|
|
|
|
|
|
|
|
m_memory =
|
|
|
|
static_cast<u8*>(std::malloc(static_cast<size_t>(RAW_SECTOR_SIZE) * static_cast<size_t>(m_memory_sectors)));
|
|
|
|
if (!m_memory)
|
|
|
|
{
|
2021-03-29 19:59:22 +00:00
|
|
|
progress->DisplayFormattedModalError("Failed to allocate memory for %u sectors", m_memory_sectors);
|
2020-07-21 14:03:07 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
progress->SetStatusText("Preloading CD image to RAM...");
|
|
|
|
progress->SetProgressRange(m_memory_sectors);
|
|
|
|
progress->SetProgressValue(0);
|
|
|
|
|
|
|
|
u8* memory_ptr = m_memory;
|
|
|
|
u32 sectors_read = 0;
|
|
|
|
for (u32 i = 0; i < image->GetIndexCount(); i++)
|
|
|
|
{
|
|
|
|
const Index& index = image->GetIndex(i);
|
|
|
|
if (index.file_sector_size == 0)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
for (u32 lba = 0; lba < index.length; lba++)
|
|
|
|
{
|
|
|
|
if (!image->ReadSectorFromIndex(memory_ptr, index, lba))
|
|
|
|
{
|
2024-05-23 10:20:16 +00:00
|
|
|
Log_ErrorFmt("Failed to read LBA {} in index {}", lba, i);
|
2020-07-21 14:03:07 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
progress->SetProgressValue(sectors_read);
|
|
|
|
memory_ptr += RAW_SECTOR_SIZE;
|
|
|
|
sectors_read++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
for (u32 i = 1; i <= image->GetTrackCount(); i++)
|
|
|
|
m_tracks.push_back(image->GetTrack(i));
|
|
|
|
|
|
|
|
u32 current_offset = 0;
|
|
|
|
for (u32 i = 0; i < image->GetIndexCount(); i++)
|
|
|
|
{
|
|
|
|
Index new_index = image->GetIndex(i);
|
|
|
|
new_index.file_index = 0;
|
|
|
|
if (new_index.file_sector_size > 0)
|
|
|
|
{
|
|
|
|
new_index.file_offset = current_offset;
|
|
|
|
current_offset += new_index.length;
|
|
|
|
}
|
|
|
|
m_indices.push_back(new_index);
|
|
|
|
}
|
|
|
|
|
|
|
|
Assert(current_offset == m_memory_sectors);
|
|
|
|
m_filename = image->GetFileName();
|
|
|
|
m_lba_count = image->GetLBACount();
|
|
|
|
|
2023-11-15 09:05:21 +00:00
|
|
|
m_sbi.LoadFromImagePath(m_filename);
|
2020-07-21 14:03:07 +00:00
|
|
|
|
|
|
|
return Seek(1, Position{0, 0, 0});
|
|
|
|
}
|
|
|
|
|
2021-03-26 16:19:23 +00:00
|
|
|
bool CDImageMemory::ReadSubChannelQ(SubChannelQ* subq, const Index& index, LBA lba_in_index)
|
2020-07-21 14:03:07 +00:00
|
|
|
{
|
2021-03-26 16:19:23 +00:00
|
|
|
if (m_sbi.GetReplacementSubChannelQ(index.start_lba_on_disc + lba_in_index, subq))
|
2020-07-21 14:03:07 +00:00
|
|
|
return true;
|
|
|
|
|
2021-03-26 16:19:23 +00:00
|
|
|
return CDImage::ReadSubChannelQ(subq, index, lba_in_index);
|
2020-07-21 14:03:07 +00:00
|
|
|
}
|
|
|
|
|
2020-12-17 17:01:57 +00:00
|
|
|
bool CDImageMemory::HasNonStandardSubchannel() const
|
|
|
|
{
|
|
|
|
return (m_sbi.GetReplacementSectorCount() > 0);
|
|
|
|
}
|
|
|
|
|
2022-07-23 03:23:54 +00:00
|
|
|
bool CDImageMemory::IsPrecached() const
|
|
|
|
{
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2020-07-21 14:03:07 +00:00
|
|
|
bool CDImageMemory::ReadSectorFromIndex(void* buffer, const Index& index, LBA lba_in_index)
|
|
|
|
{
|
|
|
|
DebugAssert(index.file_index == 0);
|
|
|
|
|
|
|
|
const u64 sector_number = index.file_offset + lba_in_index;
|
|
|
|
if (sector_number >= m_memory_sectors)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
const size_t file_offset = static_cast<size_t>(sector_number) * static_cast<size_t>(RAW_SECTOR_SIZE);
|
|
|
|
std::memcpy(buffer, &m_memory[file_offset], RAW_SECTOR_SIZE);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
std::unique_ptr<CDImage>
|
|
|
|
CDImage::CreateMemoryImage(CDImage* image, ProgressCallback* progress /* = ProgressCallback::NullProgressCallback */)
|
|
|
|
{
|
|
|
|
std::unique_ptr<CDImageMemory> memory_image = std::make_unique<CDImageMemory>();
|
|
|
|
if (!memory_image->CopyImage(image, progress))
|
|
|
|
return {};
|
|
|
|
|
|
|
|
return memory_image;
|
|
|
|
}
|