mirror of
https://github.com/RetroDECK/Duckstation.git
synced 2025-01-23 16:45:37 +00:00
302 lines
8.6 KiB
C++
302 lines
8.6 KiB
C++
// SPDX-FileCopyrightText: 2019-2022 Connor McLaughlin <stenzek@gmail.com>
|
|
// SPDX-License-Identifier: (GPL-3.0 OR CC-BY-NC-ND-4.0)
|
|
|
|
#include "iso_reader.h"
|
|
#include "cd_image.h"
|
|
#include "common/log.h"
|
|
#include <cctype>
|
|
Log_SetChannel(ISOReader);
|
|
|
|
static bool FilenamesEqual(const char* a, const char* b, u32 length)
|
|
{
|
|
u32 pos = 0;
|
|
for (; pos < length && *a != '\0' && *b != '\0'; pos++)
|
|
{
|
|
if (std::tolower(*(a++)) != std::tolower(*(b++)))
|
|
return false;
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
ISOReader::ISOReader() = default;
|
|
|
|
ISOReader::~ISOReader() = default;
|
|
|
|
bool ISOReader::Open(CDImage* image, u32 track_number)
|
|
{
|
|
m_image = image;
|
|
m_track_number = track_number;
|
|
if (!ReadPVD())
|
|
return false;
|
|
|
|
return true;
|
|
}
|
|
|
|
bool ISOReader::ReadPVD()
|
|
{
|
|
// volume descriptor start at sector 16
|
|
if (!m_image->Seek(m_track_number, 16))
|
|
return false;
|
|
|
|
// try only a maximum of 256 volume descriptors
|
|
for (u32 i = 0; i < 256; i++)
|
|
{
|
|
u8 buffer[SECTOR_SIZE];
|
|
if (m_image->Read(CDImage::ReadMode::DataOnly, 1, buffer) != 1)
|
|
return false;
|
|
|
|
const ISOVolumeDescriptorHeader* header = reinterpret_cast<ISOVolumeDescriptorHeader*>(buffer);
|
|
if (header->type_code != 1)
|
|
continue;
|
|
else if (header->type_code == 255)
|
|
break;
|
|
|
|
std::memcpy(&m_pvd, buffer, sizeof(ISOPrimaryVolumeDescriptor));
|
|
Log_DebugPrintf("PVD found at index %u", i);
|
|
return true;
|
|
}
|
|
|
|
Log_ErrorPrint("PVD not found");
|
|
return false;
|
|
}
|
|
|
|
std::optional<ISOReader::ISODirectoryEntry> ISOReader::LocateFile(const char* path)
|
|
{
|
|
u8 sector_buffer[SECTOR_SIZE];
|
|
|
|
const ISODirectoryEntry* root_de = reinterpret_cast<const ISODirectoryEntry*>(m_pvd.root_directory_entry);
|
|
if (*path == '\0' || std::strcmp(path, "/") == 0)
|
|
{
|
|
// locating the root directory
|
|
return *root_de;
|
|
}
|
|
|
|
// start at the root directory
|
|
return LocateFile(path, sector_buffer, root_de->location_le, root_de->length_le);
|
|
}
|
|
|
|
std::optional<ISOReader::ISODirectoryEntry> ISOReader::LocateFile(const char* path, u8* sector_buffer,
|
|
u32 directory_record_lba, u32 directory_record_size)
|
|
{
|
|
if (directory_record_size == 0)
|
|
{
|
|
Log_ErrorPrintf("Directory entry record size 0 while looking for '%s'", path);
|
|
return std::nullopt;
|
|
}
|
|
|
|
// strip any leading slashes
|
|
const char* path_component_start = path;
|
|
while (*path_component_start == '/' || *path_component_start == '\\')
|
|
path_component_start++;
|
|
|
|
u32 path_component_length = 0;
|
|
const char* path_component_end = path_component_start;
|
|
while (*path_component_end != '\0' && *path_component_end != '/' && *path_component_end != '\\')
|
|
{
|
|
path_component_length++;
|
|
path_component_end++;
|
|
}
|
|
|
|
// start reading directory entries
|
|
const u32 num_sectors = (directory_record_size + (SECTOR_SIZE - 1)) / SECTOR_SIZE;
|
|
if (!m_image->Seek(m_track_number, directory_record_lba))
|
|
{
|
|
Log_ErrorPrintf("Seek to LBA %u failed", directory_record_lba);
|
|
return std::nullopt;
|
|
}
|
|
|
|
for (u32 i = 0; i < num_sectors; i++)
|
|
{
|
|
if (m_image->Read(CDImage::ReadMode::DataOnly, 1, sector_buffer) != 1)
|
|
{
|
|
Log_ErrorPrintf("Failed to read LBA %u", directory_record_lba + i);
|
|
return std::nullopt;
|
|
}
|
|
|
|
u32 sector_offset = 0;
|
|
while ((sector_offset + sizeof(ISODirectoryEntry)) < SECTOR_SIZE)
|
|
{
|
|
const ISODirectoryEntry* de = reinterpret_cast<const ISODirectoryEntry*>(§or_buffer[sector_offset]);
|
|
const char* de_filename =
|
|
reinterpret_cast<const char*>(§or_buffer[sector_offset + sizeof(ISODirectoryEntry)]);
|
|
if ((sector_offset + de->entry_length) > SECTOR_SIZE || de->filename_length > de->entry_length ||
|
|
de->entry_length < sizeof(ISODirectoryEntry))
|
|
{
|
|
break;
|
|
}
|
|
|
|
sector_offset += de->entry_length;
|
|
|
|
// skip current/parent directory
|
|
if (de->filename_length == 1 && (*de_filename == '\x0' || *de_filename == '\x1'))
|
|
continue;
|
|
|
|
// check filename length
|
|
if (de->filename_length < path_component_length)
|
|
continue;
|
|
|
|
if (de->flags & ISODirectoryEntryFlag_Directory)
|
|
{
|
|
// directories don't have the version? so check the length instead
|
|
if (de->filename_length != path_component_length ||
|
|
!FilenamesEqual(de_filename, path_component_start, path_component_length))
|
|
{
|
|
continue;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
// compare filename
|
|
if (!FilenamesEqual(de_filename, path_component_start, path_component_length) ||
|
|
de_filename[path_component_length] != ';')
|
|
{
|
|
continue;
|
|
}
|
|
}
|
|
|
|
// found it. is this the file we're looking for?
|
|
if (*path_component_end == '\0')
|
|
return *de;
|
|
|
|
// if it is a directory, recurse into it
|
|
if (de->flags & ISODirectoryEntryFlag_Directory)
|
|
return LocateFile(path_component_end, sector_buffer, de->location_le, de->length_le);
|
|
|
|
// we're looking for a directory but got a file
|
|
Log_ErrorPrintf("Looking for directory but got file");
|
|
return std::nullopt;
|
|
}
|
|
}
|
|
|
|
std::string temp(path_component_start, path_component_length);
|
|
Log_ErrorPrintf("Path component '%s' not found", temp.c_str());
|
|
return std::nullopt;
|
|
}
|
|
|
|
std::vector<std::string> ISOReader::GetFilesInDirectory(const char* path)
|
|
{
|
|
std::string base_path = path;
|
|
u32 directory_record_lba;
|
|
u32 directory_record_length;
|
|
if (base_path.empty())
|
|
{
|
|
// root directory
|
|
const ISODirectoryEntry* root_de = reinterpret_cast<const ISODirectoryEntry*>(m_pvd.root_directory_entry);
|
|
directory_record_lba = root_de->location_le;
|
|
directory_record_length = root_de->length_le;
|
|
}
|
|
else
|
|
{
|
|
auto directory_de = LocateFile(base_path.c_str());
|
|
if (!directory_de)
|
|
{
|
|
Log_ErrorPrintf("Directory entry not found for '%s'", path);
|
|
return {};
|
|
}
|
|
|
|
if ((directory_de->flags & ISODirectoryEntryFlag_Directory) == 0)
|
|
{
|
|
Log_ErrorPrintf("Path '%s' is not a directory, can't list", path);
|
|
return {};
|
|
}
|
|
|
|
directory_record_lba = directory_de->location_le;
|
|
directory_record_length = directory_de->length_le;
|
|
|
|
if (base_path[base_path.size() - 1] != '/')
|
|
base_path += '/';
|
|
}
|
|
|
|
// start reading directory entries
|
|
const u32 num_sectors = (directory_record_length + (SECTOR_SIZE - 1)) / SECTOR_SIZE;
|
|
if (!m_image->Seek(m_track_number, directory_record_lba))
|
|
{
|
|
Log_ErrorPrintf("Seek to LBA %u failed", directory_record_lba);
|
|
return {};
|
|
}
|
|
|
|
std::vector<std::string> files;
|
|
u8 sector_buffer[SECTOR_SIZE];
|
|
for (u32 i = 0; i < num_sectors; i++)
|
|
{
|
|
if (m_image->Read(CDImage::ReadMode::DataOnly, 1, sector_buffer) != 1)
|
|
{
|
|
Log_ErrorPrintf("Failed to read LBA %u", directory_record_lba + i);
|
|
break;
|
|
}
|
|
|
|
u32 sector_offset = 0;
|
|
while ((sector_offset + sizeof(ISODirectoryEntry)) < SECTOR_SIZE)
|
|
{
|
|
const ISODirectoryEntry* de = reinterpret_cast<const ISODirectoryEntry*>(§or_buffer[sector_offset]);
|
|
const char* de_filename =
|
|
reinterpret_cast<const char*>(§or_buffer[sector_offset + sizeof(ISODirectoryEntry)]);
|
|
if ((sector_offset + de->entry_length) > SECTOR_SIZE || de->filename_length > de->entry_length ||
|
|
de->entry_length < sizeof(ISODirectoryEntry))
|
|
{
|
|
break;
|
|
}
|
|
|
|
sector_offset += de->entry_length;
|
|
|
|
// skip current/parent directory
|
|
if (de->filename_length == 1 && (*de_filename == '\x0' || *de_filename == '\x1'))
|
|
continue;
|
|
|
|
// strip off terminator/file version
|
|
std::string filename(de_filename, de->filename_length);
|
|
std::string::size_type pos = filename.rfind(';');
|
|
if (pos == std::string::npos)
|
|
{
|
|
Log_ErrorPrintf("Invalid filename '%s'", filename.c_str());
|
|
continue;
|
|
}
|
|
filename.erase(pos);
|
|
|
|
if (!filename.empty())
|
|
files.push_back(base_path + filename);
|
|
}
|
|
}
|
|
|
|
return files;
|
|
}
|
|
|
|
bool ISOReader::ReadFile(const char* path, std::vector<u8>* data)
|
|
{
|
|
auto de = LocateFile(path);
|
|
if (!de)
|
|
{
|
|
Log_ErrorPrintf("File not found: '%s'", path);
|
|
return false;
|
|
}
|
|
if (de->flags & ISODirectoryEntryFlag_Directory)
|
|
{
|
|
Log_ErrorPrintf("File is a directory: '%s'", path);
|
|
return false;
|
|
}
|
|
|
|
if (!m_image->Seek(m_track_number, de->location_le))
|
|
return false;
|
|
|
|
if (de->length_le == 0)
|
|
{
|
|
data->clear();
|
|
return true;
|
|
}
|
|
|
|
const u32 num_sectors = (de->length_le + (SECTOR_SIZE - 1)) / SECTOR_SIZE;
|
|
data->resize(num_sectors * u64(SECTOR_SIZE));
|
|
if (m_image->Read(CDImage::ReadMode::DataOnly, num_sectors, data->data()) != num_sectors)
|
|
return false;
|
|
|
|
data->resize(de->length_le);
|
|
return true;
|
|
}
|
|
|
|
bool ISOReader::FileExists(const char* path)
|
|
{
|
|
auto de = LocateFile(path);
|
|
return de.has_value();
|
|
}
|